<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Ujwal Jayendran</title>
    <description>The latest articles on DEV Community by Ujwal Jayendran (@lactustech).</description>
    <link>https://dev.to/lactustech</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3952999%2F789d729a-83d2-48fc-853a-75347576d14f.png</url>
      <title>DEV Community: Ujwal Jayendran</title>
      <link>https://dev.to/lactustech</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/lactustech"/>
    <language>en</language>
    <item>
      <title>How I Built a Programmatic SEO Site with 16,750 Pages Using FastAPI and PostgreSQL</title>
      <dc:creator>Ujwal Jayendran</dc:creator>
      <pubDate>Tue, 26 May 2026 17:31:40 +0000</pubDate>
      <link>https://dev.to/lactustech/how-i-built-a-programmatic-seo-site-with-16750-pages-using-fastapi-and-postgresql-33ji</link>
      <guid>https://dev.to/lactustech/how-i-built-a-programmatic-seo-site-with-16750-pages-using-fastapi-and-postgresql-33ji</guid>
      <description>&lt;p&gt;&lt;em&gt;A deep dive into building BSBFinder.com — a free Australian BSB number lookup tool — from data pipeline to deployment.&lt;/em&gt;&lt;/p&gt;




&lt;p&gt;Every Australian bank transfer requires a BSB number — a 6-digit code identifying the bank and branch. There are over 16,750 active BSB codes, and the existing lookup tools were either clunky, ad-riddled, or buried inside bank websites.&lt;/p&gt;

&lt;p&gt;So I built &lt;a href="https://bsbfinder.com" rel="noopener noreferrer"&gt;BSBFinder.com&lt;/a&gt; — a fast, free tool that lets you search any BSB code and get the bank name, branch address, SWIFT code, and payment capabilities instantly.&lt;/p&gt;

&lt;p&gt;Here's how I built it.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Stack
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Backend:&lt;/strong&gt; FastAPI (Python)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Database:&lt;/strong&gt; PostgreSQL&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Templating:&lt;/strong&gt; Jinja2 (server-side rendered)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Reverse Proxy:&lt;/strong&gt; Caddy&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Infrastructure:&lt;/strong&gt; Docker on AWS Lightsail&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Data Source:&lt;/strong&gt; AusPayNet (official BSB registry)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;I chose server-side rendering over a SPA framework deliberately. For a programmatic SEO site with 16,750+ individual pages, SSR gives you crawlable HTML that search engines can index immediately — no JavaScript rendering required.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Data Pipeline
&lt;/h2&gt;

&lt;p&gt;The foundation of the entire project is the BSB dataset from AusPayNet, the official body that manages BSB allocations in Australia.&lt;/p&gt;

&lt;p&gt;The pipeline works like this:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Extract&lt;/strong&gt; — Parse the official BSB data file (CSV format with bank codes, branch names, addresses, states, postcodes, and payment method flags)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Enrich&lt;/strong&gt; — Add SWIFT/BIC codes by mapping BSB prefixes to their parent bank's international codes&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Snapshot&lt;/strong&gt; — Store timestamped snapshots to track historical changes (branch closures, mergers, relocations)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Load&lt;/strong&gt; — Upsert into PostgreSQL with proper indexing&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The snapshot system is one of the features I'm most proud of. Banks merge, branches close, and BSB codes get reassigned. By storing periodic snapshots, BSBFinder can show users when a BSB was last changed and what changed — useful for anyone dealing with an old BSB that no longer works.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight python"&gt;&lt;code&gt;&lt;span class="c1"&gt;# Simplified snapshot comparison logic
&lt;/span&gt;&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;detect_changes&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;current_snapshot&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;previous_snapshot&lt;/span&gt;&lt;span class="p"&gt;):&lt;/span&gt;
    &lt;span class="n"&gt;changes&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;[]&lt;/span&gt;
    &lt;span class="k"&gt;for&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;data&lt;/span&gt; &lt;span class="ow"&gt;in&lt;/span&gt; &lt;span class="n"&gt;current_snapshot&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;items&lt;/span&gt;&lt;span class="p"&gt;():&lt;/span&gt;
        &lt;span class="k"&gt;if&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt; &lt;span class="ow"&gt;not&lt;/span&gt; &lt;span class="ow"&gt;in&lt;/span&gt; &lt;span class="n"&gt;previous_snapshot&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
            &lt;span class="n"&gt;changes&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;append&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;bsb&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;type&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;added&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;})&lt;/span&gt;
        &lt;span class="k"&gt;elif&lt;/span&gt; &lt;span class="n"&gt;data&lt;/span&gt; &lt;span class="o"&gt;!=&lt;/span&gt; &lt;span class="n"&gt;previous_snapshot&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;]:&lt;/span&gt;
            &lt;span class="n"&gt;changes&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;append&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;bsb&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;type&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;modified&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; 
                          &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;old&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="n"&gt;previous_snapshot&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;],&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;new&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="n"&gt;data&lt;/span&gt;&lt;span class="p"&gt;})&lt;/span&gt;

    &lt;span class="k"&gt;for&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt; &lt;span class="ow"&gt;in&lt;/span&gt; &lt;span class="n"&gt;previous_snapshot&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
        &lt;span class="k"&gt;if&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt; &lt;span class="ow"&gt;not&lt;/span&gt; &lt;span class="ow"&gt;in&lt;/span&gt; &lt;span class="n"&gt;current_snapshot&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
            &lt;span class="n"&gt;changes&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;append&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;bsb&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="n"&gt;bsb&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;type&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;discontinued&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;})&lt;/span&gt;

    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="n"&gt;changes&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Programmatic SEO: 16,750 Pages from Templates
&lt;/h2&gt;

&lt;p&gt;The core SEO strategy is programmatic — each BSB code gets its own page generated from a template. But "programmatic" doesn't mean "thin." Each BSB page includes:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;The BSB number, bank name, and full branch address&lt;/li&gt;
&lt;li&gt;SWIFT/BIC code for the parent bank&lt;/li&gt;
&lt;li&gt;Payment method support (BECS, NPP, Direct Entry)&lt;/li&gt;
&lt;li&gt;Nearby branches from the same bank&lt;/li&gt;
&lt;li&gt;A FAQ section with the most common questions for that specific BSB&lt;/li&gt;
&lt;li&gt;Schema.org structured data for rich search results&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The key lesson I learned: &lt;strong&gt;Google treats 16,750 near-identical pages very differently from 16,750 pages that each have unique, useful content.&lt;/strong&gt; The nearby branches section, the dynamic FAQ, and the payment capability details make each page genuinely different.&lt;/p&gt;

&lt;h3&gt;
  
  
  URL Structure
&lt;/h3&gt;

&lt;p&gt;I went with a flat, readable structure:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;/bsb/062-000    → Individual BSB page
/bank/cba       → All BSBs for Commonwealth Bank
/state/nsw      → All BSBs in New South Wales
/suburb/vic/melbourne → BSBs in Melbourne, VIC
/postcode/2000  → BSBs in postcode 2000
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This creates natural internal linking — each BSB page links to its bank page, state page, suburb page, and postcode page, creating a deep web of interconnections that search engines love.&lt;/p&gt;

&lt;h2&gt;
  
  
  Sitemaps at Scale
&lt;/h2&gt;

&lt;p&gt;With 16,750 BSB pages plus bank pages, state pages, suburb pages, postcode pages, and guide articles, the total URL count exceeds 20,000. Google's sitemap limit is 50,000 URLs per file, but I split them into batches of 5,000 for better crawl management.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;/sitemap-index.xml
  ├── /sitemap-bsb-1.xml     (5,000 BSB pages)
  ├── /sitemap-bsb-2.xml     (5,000 BSB pages)
  ├── /sitemap-bsb-3.xml     (5,000 BSB pages)
  ├── /sitemap-bsb-4.xml     (remaining BSB pages)
  ├── /sitemap-banks.xml      (bank pages)
  ├── /sitemap-locations.xml  (state, suburb, postcode pages)
  └── /sitemap-guides.xml     (editorial content)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;I submitted these in batches over several days rather than all at once — it helps avoid overwhelming the crawler and triggering spam flags.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Tools
&lt;/h2&gt;

&lt;p&gt;Beyond the core lookup, I built several tools that add genuine utility:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;BSB Validator&lt;/strong&gt; — Verify a BSB number is valid before making a transfer (checks format, prefix validity, and whether the BSB is still active)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Bulk Lookup&lt;/strong&gt; — Upload a CSV of BSB numbers and get all details back in one go (useful for payroll and accounting)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;BSB Decoder&lt;/strong&gt; — Break down what each digit means (first 2 = bank, next 1 = state, last 3 = branch)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Branch Compare&lt;/strong&gt; — Side-by-side comparison of two branches&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;SWIFT Lookup&lt;/strong&gt; — Find the SWIFT/BIC code for any Australian bank&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Each tool has its own page with its own SEO value. The validator alone generates meaningful search traffic for queries like "check BSB number" and "BSB validator."&lt;/p&gt;

&lt;h2&gt;
  
  
  Performance Considerations
&lt;/h2&gt;

&lt;p&gt;For a site that serves 16,750+ unique pages, performance matters. Here's what I optimized:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Database queries:&lt;/strong&gt; Every BSB lookup is a simple primary key query — O(1) with PostgreSQL's B-tree index. Response times are under 5ms for any BSB lookup.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Caddy as reverse proxy:&lt;/strong&gt; Caddy handles TLS termination, HTTP/2, and automatic HTTPS. It also serves static assets directly without hitting the FastAPI backend.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Docker deployment:&lt;/strong&gt; The entire stack runs in Docker containers on a single AWS Lightsail instance ($5/month). For the traffic levels of a niche utility site, this is more than sufficient.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="c1"&gt;# docker-compose.yml (simplified)&lt;/span&gt;
&lt;span class="na"&gt;services&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;web&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;build&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;.&lt;/span&gt;
    &lt;span class="na"&gt;ports&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;8000:8000"&lt;/span&gt;
    &lt;span class="na"&gt;depends_on&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;db&lt;/span&gt;
  &lt;span class="na"&gt;db&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;postgres:15&lt;/span&gt;
    &lt;span class="na"&gt;volumes&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;pgdata:/var/lib/postgresql/data&lt;/span&gt;
  &lt;span class="na"&gt;caddy&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;caddy:2&lt;/span&gt;
    &lt;span class="na"&gt;ports&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;80:80"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;443:443"&lt;/span&gt;
    &lt;span class="na"&gt;volumes&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;./Caddyfile:/etc/caddy/Caddyfile&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Free REST API
&lt;/h2&gt;

&lt;p&gt;I also built a free REST API for developers who need BSB data programmatically. It supports single lookups, search, and bank/branch listing — all without authentication for reasonable usage.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# Look up a BSB&lt;/span&gt;
curl https://bsbfinder.com/api/bsb/062-000

&lt;span class="c"&gt;# Search by bank name&lt;/span&gt;
curl https://bsbfinder.com/api/search?q&lt;span class="o"&gt;=&lt;/span&gt;commonwealth+sydney
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This serves a dual purpose: it's genuinely useful for developers building Australian fintech products, and it creates an incentive for technical blogs and documentation to link to BSBFinder as a data source.&lt;/p&gt;

&lt;h2&gt;
  
  
  Lessons Learned
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;1. Server-side rendering wins for programmatic SEO.&lt;/strong&gt; SPA frameworks add unnecessary complexity when your primary goal is search engine indexing. Jinja2 templates are simple, fast, and produce crawlable HTML.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;2. Content depth matters more than content volume.&lt;/strong&gt; 16,750 thin pages will get flagged. 16,750 pages that each answer a specific question with unique data will get indexed and ranked.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;3. Sitemaps are a conversation with search engines.&lt;/strong&gt; Don't dump 20,000 URLs at once. Submit in batches, monitor indexing rates, and expand as trust builds.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;4. Internal linking is your most powerful on-page SEO tool.&lt;/strong&gt; Every BSB page links to its bank, state, suburb, and postcode pages. Every bank page links to all its branches. This creates a dense link graph that distributes authority across the entire site.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;5. Build tools, not just pages.&lt;/strong&gt; The validator, bulk lookup, and decoder tools generate their own search traffic and give users a reason to bookmark the site.&lt;/p&gt;

&lt;h2&gt;
  
  
  What's Next
&lt;/h2&gt;

&lt;p&gt;I'm tracking historical BSB changes to build a "discontinued BSBs" section — useful for anyone trying to figure out where an old BSB number was redirected. I'm also exploring a premium API tier for high-volume users.&lt;/p&gt;

&lt;p&gt;If you're building a programmatic SEO site, the key takeaway is this: &lt;strong&gt;every page should answer a question that someone is actually searching for.&lt;/strong&gt; If it doesn't, it's filler — and search engines can tell.&lt;/p&gt;




&lt;p&gt;Check out &lt;a href="https://bsbfinder.com" rel="noopener noreferrer"&gt;BSBFinder.com&lt;/a&gt; if you're curious, or hit the &lt;a href="https://bsbfinder.com/api" rel="noopener noreferrer"&gt;API docs&lt;/a&gt; if you want to build something with it.&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Questions? Drop a comment below — happy to dive deeper into any part of the stack.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>python</category>
      <category>seo</category>
      <category>fastapi</category>
    </item>
  </channel>
</rss>
