<?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: Julio Cesar </title>
    <description>The latest articles on DEV Community by Julio Cesar  (@jcvanz).</description>
    <link>https://dev.to/jcvanz</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%2F3906678%2F9bb0e716-72b1-4195-84df-74dc3c1fc597.jpeg</url>
      <title>DEV Community: Julio Cesar </title>
      <link>https://dev.to/jcvanz</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/jcvanz"/>
    <language>en</language>
    <item>
      <title>How I built a 13-tool Micro-SaaS with $0 server costs using React and Web APIs</title>
      <dc:creator>Julio Cesar </dc:creator>
      <pubDate>Thu, 30 Apr 2026 20:13:36 +0000</pubDate>
      <link>https://dev.to/jcvanz/how-i-built-a-13-tool-micro-saas-with-0-server-costs-using-react-and-web-apis-27fh</link>
      <guid>https://dev.to/jcvanz/how-i-built-a-13-tool-micro-saas-with-0-server-costs-using-react-and-web-apis-27fh</guid>
      <description>&lt;p&gt;If you've ever tried building a SaaS or a tool aggregator, you know the drill: file uploads equal server costs. Processing PDFs or stripping image backgrounds usually requires setting up a backend, managing storage (like AWS S3), and dealing with potential privacy liabilities.&lt;/p&gt;

&lt;p&gt;I was tired of uploading my sensitive files to random "free" tools that hit me with paywalls or forced me to create accounts. So, I challenged myself: &lt;strong&gt;Could I build a comprehensive toolbox where every single task runs 100% in the user's browser?&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The answer is yes.&lt;/p&gt;

&lt;p&gt;I built &lt;a href="https://www.myzerotools.online/" rel="noopener noreferrer"&gt;ZeroTools&lt;/a&gt;, a collection of 13 everyday utilities (Image Compressor, Background Remover, PDF Optimizer, Code Formatters, etc.) &lt;strong&gt;with a strict Zero-Backend architecture.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Here is how I managed to keep my server costs at exactly $0 while guaranteeing total privacy for the users.&lt;/p&gt;

&lt;p&gt;🛠️ &lt;strong&gt;The Architecture: React + Vite + Vercel&lt;/strong&gt;&lt;br&gt;
The foundation is simple. The entire project is a static React application built with Vite and hosted on Vercel's free hobby tier. Since there is no Node.js backend or database, Vercel just serves the static assets globally via their CDN.&lt;/p&gt;

&lt;p&gt;But the real magic happens in how I used modern Web APIs to replace backend servers.&lt;/p&gt;

&lt;p&gt;🖼️ &lt;strong&gt;1. Image Compression using HTML5 Canvas&lt;/strong&gt;&lt;br&gt;
Instead of uploading images to a server to run ImageMagick or Sharp, I used the native browser  API.&lt;/p&gt;

&lt;p&gt;When a user selects an image, the browser reads it via FileReader, draws it onto an invisible canvas, and then exports it at a lower quality or different format.&lt;/p&gt;

&lt;p&gt;Here is a simplified version of the logic:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Read the file locally&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;reader&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;FileReader&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;span class="nx"&gt;reader&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;onload&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;e&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&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;img&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Image&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
  &lt;span class="nx"&gt;img&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;src&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;e&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;target&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;result&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

  &lt;span class="nx"&gt;img&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;onload&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="c1"&gt;// Draw on a temporary canvas&lt;/span&gt;
    &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;canvas&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nb"&gt;document&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;createElement&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;canvas&lt;/span&gt;&lt;span class="dl"&gt;'&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;ctx&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;canvas&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;getContext&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;2d&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
    &lt;span class="nx"&gt;canvas&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;width&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;img&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;width&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
    &lt;span class="nx"&gt;canvas&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;height&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;img&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;height&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
    &lt;span class="nx"&gt;ctx&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;drawImage&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;img&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="mi"&gt;0&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
    &lt;span class="c1"&gt;// Export as compressed WebP (0.7 = 70% quality)&lt;/span&gt;
    &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;compressedDataUrl&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;canvas&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;toDataURL&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;image/webp&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="mf"&gt;0.7&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;

    &lt;span class="c1"&gt;// Now the user can download the compressedDataUrl directly!&lt;/span&gt;
  &lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="nx"&gt;reader&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;readAsDataURL&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;file&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;🪄 &lt;strong&gt;2. AI Background Removal via WebAssembly (WASM)&lt;/strong&gt;&lt;br&gt;
This was the most exciting part. Usually, removing backgrounds requires an expensive Python/PyTorch backend API.&lt;/p&gt;

&lt;p&gt;Instead, I used &lt;a class="mentioned-user" href="https://dev.to/imgly"&gt;@imgly&lt;/a&gt;/background-removal, which ports a machine learning model to WebAssembly. The first time a user opens the tool, their browser downloads a tiny AI model (~40MB, which is cached for future visits) and executes the neural network locally using their own device's CPU/GPU!&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="nx"&gt;imglyRemoveBackground&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;@imgly/background-removal&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;async&lt;/span&gt; &lt;span class="kd"&gt;function&lt;/span&gt; &lt;span class="nf"&gt;removeBg&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;imageFile&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// The AI runs completely inside the user's browser via WASM&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;imageBlob&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;imglyRemoveBackground&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;imageFile&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;url&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;URL&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;createObjectURL&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;imageBlob&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="nx"&gt;url&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;&lt;em&gt;Zero server cost, infinite scaling, and the user's photos never touch the internet.&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;📄 &lt;strong&gt;3. PDF Manipulation with pdf-lib&lt;/strong&gt;&lt;br&gt;
For the PDF Compressor, I utilized the pdf-lib library. It allows you to load, modify, and save PDFs directly in JavaScript. By reading the local file buffer, I can optimize the document structure and strip unnecessary metadata without ever sending the document over the network.&lt;/p&gt;

&lt;p&gt;🚀 &lt;strong&gt;The Result&lt;/strong&gt;&lt;br&gt;
By shifting the compute power from my servers to the user's local device, the result is a blazing-fast, infinitely scalable Micro-SaaS.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;- Privacy:&lt;/strong&gt; 100% guaranteed.&lt;br&gt;
&lt;strong&gt;- Limits:&lt;/strong&gt; None. Users can process 1,000 images if they want; it only costs them their own electricity.&lt;br&gt;
&lt;strong&gt;- Hosting cost:&lt;/strong&gt; $0.&lt;br&gt;
If you want to test the speed and see the tools in action, check it out here: &lt;a href="https://www.myzerotools.online/" rel="noopener noreferrer"&gt;ZeroTools&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;I’d love to hear your thoughts on this architecture! Have you built any client-side-only tools recently? Drop your feedback or any edge cases you find in the comments.&lt;/p&gt;

&lt;p&gt;Access via the link: &lt;a href="https://www.myzerotools.online/" rel="noopener noreferrer"&gt;ZeroTools&lt;/a&gt;&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>react</category>
      <category>javascript</category>
      <category>showdev</category>
    </item>
  </channel>
</rss>
