<?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: Anirudha Sonwane</title>
    <description>The latest articles on DEV Community by Anirudha Sonwane (@anirudha_sonwane_ca3fc720).</description>
    <link>https://dev.to/anirudha_sonwane_ca3fc720</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.us-east-2.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F4002203%2Fded1a1e7-bc6c-400e-b920-a10e4852f5ce.png</url>
      <title>DEV Community: Anirudha Sonwane</title>
      <link>https://dev.to/anirudha_sonwane_ca3fc720</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/anirudha_sonwane_ca3fc720"/>
    <language>en</language>
    <item>
      <title>I built a free online toolbox with 260+ tools — here's the tech stack and what I learned</title>
      <dc:creator>Anirudha Sonwane</dc:creator>
      <pubDate>Fri, 26 Jun 2026 06:45:15 +0000</pubDate>
      <link>https://dev.to/anirudha_sonwane_ca3fc720/i-built-a-free-online-toolbox-with-260-tools-heres-the-tech-stack-and-what-i-learned-2o44</link>
      <guid>https://dev.to/anirudha_sonwane_ca3fc720/i-built-a-free-online-toolbox-with-260-tools-heres-the-tech-stack-and-what-i-learned-2o44</guid>
      <description>&lt;p&gt;Every small task used to mean a new tab. JSON formatter on one site, GST calculator on another, PDF merger somewhere that wanted my email before it would merge two pages. Ads everywhere, slow UIs, and that low-grade worry about uploading a payslip or invoice to a server I do not control. I got tired of juggling twenty bookmarks for work that should take thirty seconds — so I started building one place for all of it.&lt;/p&gt;

&lt;h2&gt;
  
  
  What ToolReign is
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://toolreign.com" rel="noopener noreferrer"&gt;ToolReign&lt;/a&gt; is a free online toolbox: &lt;strong&gt;260+ utilities across 15 categories&lt;/strong&gt;, all running in your browser. Developer tools (JSON formatter, JWT decoder, API client), text utilities, SEO helpers, PDF and image tools, spreadsheets, and a finance section I built with India in mind — GST with CGST/SGST/IGST splits, EMI and SIP calculators, HRA exemption, gratuity, income tax estimates, and more.&lt;/p&gt;

&lt;p&gt;The idea is straightforward: open a tool, do the work, leave. No signup wall, no file uploads to a backend, no account to manage. I am &lt;a href="https://www.linkedin.com/in/anirudha-sonwane/" rel="noopener noreferrer"&gt;Anirudha Sonwane&lt;/a&gt;, a Senior Software Engineer at Giant Leap Systems in Pune. ToolReign is a side project I build around my day job — not a pitch deck, just something I wished existed.&lt;/p&gt;

&lt;h2&gt;
  
  
  The tech stack decisions
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Next.js 14 App Router and static export
&lt;/h3&gt;

&lt;p&gt;Each tool lives at its own route under &lt;code&gt;src/app/{category}/{tool-slug}/&lt;/code&gt;. That maps cleanly to SEO: one URL, one search intent, one page of metadata. The site exports statically (&lt;code&gt;output: 'export'&lt;/code&gt;), so production deployment is uploading an &lt;code&gt;out/&lt;/code&gt; folder to static hosting — no Node server to babysit.&lt;/p&gt;

&lt;p&gt;The App Router made this scale. Add a page component, register the slug in &lt;code&gt;tool-registry.json&lt;/code&gt;, and the sitemap, category hubs, and search index pick it up automatically. At 260+ tools, hand-maintaining URLs would have broken within a month.&lt;/p&gt;

&lt;h3&gt;
  
  
  100% client-side — the decision that shaped everything
&lt;/h3&gt;

&lt;p&gt;This was the core architectural bet, and it is also the privacy story: &lt;strong&gt;your data never leaves the browser.&lt;/strong&gt; Finance calculators are plain TypeScript math with &lt;code&gt;useMemo&lt;/code&gt;. PDF merge and split use &lt;strong&gt;pdf-lib&lt;/strong&gt; and &lt;strong&gt;pdfjs-dist&lt;/strong&gt; in-tab. Images go through the Canvas API. Audio through the Web Audio API.&lt;/p&gt;

&lt;p&gt;The engineering is harder than the marketing. A five-file PDF merge on a mid-range phone can freeze the main thread if you are careless. I had to think about memory when combining multi-megabyte documents, show progress where it matters, and avoid loading entire files when a page-range operation would do. Getting "merge these PDFs" to feel instant without a server was more interesting than wiring another REST endpoint.&lt;/p&gt;

&lt;h3&gt;
  
  
  The supporting cast
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Tailwind CSS&lt;/strong&gt; and &lt;strong&gt;next-themes&lt;/strong&gt; for light, dark, and system mode — calculators get used late at night more than I expected.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Recharts&lt;/strong&gt; for finance visualisations: EMI principal-vs-interest donuts, SIP growth curves.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;DOMPurify&lt;/strong&gt; anywhere user-adjacent HTML might render — small surface area, non-negotiable.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Security headers&lt;/strong&gt; in &lt;code&gt;next.config.js&lt;/code&gt; — CSP, HSTS, &lt;code&gt;X-Frame-Options&lt;/code&gt;, a tight Permissions-Policy. Static sites still get scanned.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;PWA&lt;/strong&gt; via &lt;code&gt;next-pwa&lt;/code&gt; so repeat visitors can install the toolbox like an app.&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  One scaling problem: metadata without drift
&lt;/h3&gt;

&lt;p&gt;Early on I realised I could not hand-write Open Graph tags for hundreds of tools. Everything funnels through a shared helper:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// src/lib/seo.ts (simplified)&lt;/span&gt;
&lt;span class="k"&gt;export&lt;/span&gt; &lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;buildMetadata&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt; &lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;path&lt;/span&gt; &lt;span class="p"&gt;}:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="nl"&gt;title&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;
  &lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;
  &lt;span class="nx"&gt;path&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kr"&gt;string&lt;/span&gt;
&lt;span class="p"&gt;}):&lt;/span&gt; &lt;span class="nx"&gt;Metadata&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;canonical&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;SITE&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;url&lt;/span&gt;&lt;span class="p"&gt;}${&lt;/span&gt;&lt;span class="nx"&gt;path&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;replace&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="sr"&gt;/&lt;/span&gt;&lt;span class="se"&gt;([^/])&lt;/span&gt;&lt;span class="sr"&gt;$/&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;$1/&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;title&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;default&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt; | &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;SITE&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;template&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="s2"&gt;`%s | &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;SITE&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;name&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
    &lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;keywords&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;extractKeywords&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;),&lt;/span&gt;
    &lt;span class="na"&gt;openGraph&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;type&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;website&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;url&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;canonical&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;description&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;locale&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;en_IN&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
    &lt;span class="na"&gt;twitter&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;card&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;summary_large_image&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;title&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;description&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
    &lt;span class="na"&gt;alternates&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;canonical&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Each tool page calls &lt;code&gt;buildMetadata()&lt;/code&gt; once. Canonical URLs, Twitter cards, and &lt;code&gt;en_IN&lt;/code&gt; locale stay consistent. The sitemap reads from the same &lt;code&gt;tool-registry.json&lt;/code&gt;, so search indexing and on-site navigation never disagree about what exists.&lt;/p&gt;

&lt;h2&gt;
  
  
  What surprised me
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;SEO is half the product.&lt;/strong&gt; I assumed I would spend ninety percent of my time on calculator logic. Reality: JSON-LD per tool (&lt;code&gt;WebApplication&lt;/code&gt;, FAQ schema, HowTo steps), breadcrumb markup, related-tool links, synonym keywords for long-tail queries, collapsible formula explainers on finance pages. Google's AI overviews love "how is EMI calculated?" — a tool site competes on clarity as much as code.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;India-specific finance tools punched above their weight.&lt;/strong&gt; Global EMI calculators are everywhere. Adding GST split logic, HRA exemption rules, and lakh/crore formatting with &lt;code&gt;en-IN&lt;/code&gt; locale matched how people actually search here. Locale is not polish — it is a product decision.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Client-side PWAs are more capable than I assumed.&lt;/strong&gt; Installable shell, offline-friendly caching, no account — it fits the "I just need to convert this PDF" use case better than a bloated web app.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;What I would do differently:&lt;/strong&gt; invest in a stricter component scaffold earlier — shared input toolbars, validation helpers, result placeholders, WhatsApp share buttons, formula explainers — before the finance silo grew past twenty calculators. Copy-paste refactors across similar tools cost more time than the shared abstractions saved later.&lt;/p&gt;

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

&lt;p&gt;I am continuing down the long tail: more text and developer utilities, deeper finance coverage, and better mobile UX on canvas-heavy tools like handwriting generators. AdSense is on the horizon, but the constraint stays the same — do not interrupt the work surface.&lt;/p&gt;

&lt;p&gt;If you try a tool and something feels off, I genuinely want to hear it. Comment here or use the contact form on the site. Broken on Safari? Confusing GST inclusive vs exclusive? Tell me.&lt;/p&gt;

&lt;h2&gt;
  
  
  Closing
&lt;/h2&gt;

&lt;p&gt;ToolReign started as "one bookmark instead of twenty" and turned into a lesson in static SEO at scale, in-browser media processing, and building for a specific audience without locking everyone else out. It is a side project built after hours in Pune, alongside my work at Giant Leap Systems.&lt;/p&gt;

&lt;p&gt;Happy to answer questions about the stack, static export tradeoffs, or how to structure a registry when your tool count stops being cute.&lt;/p&gt;

&lt;p&gt;Check it out at &lt;a href="https://toolreign.com" rel="noopener noreferrer"&gt;toolreign.com&lt;/a&gt; — completely free, no account needed.&lt;/p&gt;

</description>
      <category>ai</category>
      <category>nextjs</category>
      <category>webdev</category>
      <category>programming</category>
    </item>
  </channel>
</rss>
