Next.jsとFirebaseを使ってインドの小さな町向けのハイパーローカルクーポンプラットフォームを構築した方法
原題: How I Built a Hyperlocal Coupon Platform with Next.js Firebase for Small-Town India
分析結果
- カテゴリ
- IT
- 重要度
- 56
- トレンドスコア
- 18
- 要約
- この記事では、インドの小さな町向けにハイパーローカルクーポンプラットフォームをNext.jsとFirebaseを用いて構築した経験を紹介しています。プロジェクトの背景、技術的な選択、開発プロセス、直面した課題、そしてそれらをどのように克服したかについて詳しく説明しています。地域のビジネスを支援し、ユーザーに特化したクーポンを提供することを目指した取り組みが強調されています。
- キーワード
The Problem Small local shops in Tier-2/3 Indian cities (think Tirunelveli, Tenkasi, Dindigul) have zero digital presence . No website, no app, no way to offer digital coupons. National platforms like Nearbuy and MagicPin don't serve these towns. I'm a developer at Blumensoft Technologies , and we built goCoupon — a hyperlocal coupon platform where local shops can create digital coupon codes, and shoppers can claim them for free. Tech Stack Layer Technology Frontend * Next.js * Database Firebase Firestore Auth Firebase Auth Hosting Google Cloud Run, Firebase App Hosting Mobile App Flutter Styling Tailwind CSS Why Next.js 14? 1. SEO is Everything for Us We need Google to index pages like /offers/tirunelveli and /restaurants-near-me . Next.js gives us: Server-Side Rendering (SSR) — Google sees fully rendered HTML Incremental Static Regeneration (ISR) — Pages rebuild every hour without redeploying Dynamic metadata — Each city page gets unique title, description, and structured data // Every city page gets unique SEO metadata export async function generateMetadata ({ params }: Props ): Promise < Metadata > { const city = getSEOCity ( params . city ); return { title : `Best Offers in ${ city . name } Today | Deals & Coupons` , description : `Find ${ city . name } offers...` , }; } 2. App Router + React Server Components Server Components let us fetch Firestore data without sending the Firebase SDK to the browser. Less JavaScript = faster pages = better Core Web Vitals. 3. Built-in Sitemap & Robots Next.js generates sitemap.xml and robots.txt dynamically: // sitemap.ts — auto-generates XML sitemap export default async function sitemap () { const cities = await getActiveTerritories (); return cities . map ( city => ({ url : `https://gocoupon.in/offers/ ${ city } ` , lastModified : new Date (), changeFrequency : ' daily ' , })); } The SEO Strategy That Worked We created 16 SEO landing pages targeting real search queries: Page Target Keyword Monthly Searches /today-offers today offers near me 25,000+ /free-coupons free coupon code 30,000+ /restaurants-near-me restaurant offers near me 15,000+ /salons-near-me salon offers near me 8,000+ /best-deals best deals today 10,000+ Each page has: ✅ JSON-LD structured data (BreadcrumbList, FAQPage) ✅ Unique H1 with target keyword ✅ FAQ section (targets Google featured snippets) ✅ Internal linking to city pages Firebase Architecture Firestore Structure: ├── businesses/ │ ├── {businessId} │ │ ├── name, category, city │ │ └── offers/ (subcollection) │ │ └── {offerId} │ │ ├── title, discount, validFrom, validThrough │ │ └── couponCodes/ (subcollection) │ │ └── {codeId} — unique per user Key design decision: Every coupon code is unique and non-shareable. When a user claims a coupon, we generate a unique code tied to their user ID. This prevents abuse and gives businesses real tracking. The IST Timezone Trap 🕐 Our Cloud Run server runs on UTC . But our users are in IST (UTC+5:30) . An offer expiring "today" (June 26 IST) was showing as expired at 6:30 PM IST because the server compared against UTC midnight. Fix: export function isExpiredIST ( validThrough : string ): boolean { const nowIST = new Date ( Date . now () + 5.5 * 60 * 60 * 1000 ); const endIST = new Date ( validThrough + ' T23:59:59+05:30 ' ); return nowIST > endIST ; } Results So Far 🏙️ Live across 38 districts in Tamil Nadu 📄 20+ indexed pages on Google (from ~5 before) 🎟️ Unique coupon codes — no sharing, no abuse 📱 Flutter mobile app for Android & iOS Try It 👉 gocoupon.in — Browse offers near you If you're building for local/hyperlocal markets, I'd love to hear your approach. Drop a comment! I'm Sakthivel, developer at Blumensoft Technologies, Tirunelveli. We build digital products for small-town India. The Problem Small local shops in Tier-2/3 Indian cities (think Tirunelveli, Tenkasi, Dindigul) have zero digital presence . No website, no app, no way to offer digital coupons. National platforms like Nearbuy and MagicPin don't serve these towns. I'm a developer at Blumensoft Technologies , and we built goCoupon — a hyperlocal coupon platform where local shops can create digital coupon codes, and shoppers can claim them for free. Tech Stack Layer Technology Frontend * Next.js * Database Firebase Firestore Auth Firebase Auth Hosting Google Cloud Run, Firebase App Hosting Mobile App Flutter Styling Tailwind CSS Why Next.js 14? 1. SEO is Everything for Us We need Google to index pages like /offers/tirunelveli and /restaurants-near-me . Next.js gives us: Server-Side Rendering (SSR) — Google sees fully rendered HTML Incremental Static Regeneration (ISR) — Pages rebuild every hour without redeploying Dynamic metadata — Each city page gets unique title, description, and structured data // Every city page gets unique SEO metadata export async function generateMetadata ({ params }: Props ): Promise < Metadata > { const city = getSEOCity ( params . city ); return { title : `Best Offers in ${ city . name } Today | Deals & Coupons` , description : `Find ${ city . name } offers...` , }; } 2. App Router + React Server Components Server Components let us fetch Firestore data without sending the Firebase SDK to the browser. Less JavaScript = faster pages = better Core Web Vitals. 3. Built-in Sitemap & Robots Next.js generates sitemap.xml and robots.txt dynamically: // sitemap.ts — auto-generates XML sitemap export default async function sitemap () { const cities = await getActiveTerritories (); return cities . map ( city => ({ url : `https://gocoupon.in/offers/ ${ city } ` , lastModified : new Date (), changeFrequency : ' daily ' , })); } The SEO Strategy That Worked We created 16 SEO landing pages targeting real search queries: Page Target Keyword Monthly Searches /today-offers today offers near me 25,000+ /free-coupons free coupon code 30,000+ /restaurants-near-me restaurant offers near me 15,000+ /salons-near-me salon offers near me 8,000+ /best-deals best deals today 10,000+ Each page has: ✅ JSON-LD structured data (BreadcrumbList, FAQPage) ✅ Unique H1 with target keyword ✅ FAQ section (targets Google featured snippets) ✅ Internal linking to city pages Firebase Architecture Firestore Structure: ├── businesses/ │ ├── {businessId} │ │ ├── name, category, city │ │ └── offers/ (subcollection) │ │ └── {offerId} │ │ ├── title, discount, validFrom, validThrough │ │ └── couponCodes/ (subcollection) │ │ └── {codeId} — unique per user Key design decision: Every coupon code is unique and non-shareable. When a user claims a coupon, we generate a unique code tied to their user ID. This prevents abuse and gives businesses real tracking. The IST Timezone Trap 🕐 Our Cloud Run server runs on UTC . But our users are in IST (UTC+5:30) . An offer expiring "today" (June 26 IST) was showing as expired at 6:30 PM IST because the server compared against UTC midnight. Fix: export function isExpiredIST ( validThrough : string ): boolean { const nowIST = new Date ( Date . now () + 5.5 * 60 * 60 * 1000 ); const endIST = new Date ( validThrough + ' T23:59:59+05:30 ' ); return nowIST > endIST ; } Results So Far 🏙️ Live across 38 districts in Tamil Nadu 📄 20+ indexed pages on Google (from ~5 before) 🎟️ Unique coupon codes — no sharing, no abuse 📱 Flutter mobile app for Android & iOS Try It 👉 gocoupon.in — Browse offers near you If you're building for local/hyperlocal markets, I'd love to hear your approach. Drop a comment! I'm Sakthivel, developer at Blumensoft Technologies, Tirunelveli. We build digital products for small-town India.