<?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>Forem: muhammad naveed</title>
    <description>The latest articles on Forem by muhammad naveed (@muhammad-naveed).</description>
    <link>https://forem.com/muhammad-naveed</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%2F2953554%2F327ed248-1d02-47f2-9d35-202df04f0836.png</url>
      <title>Forem: muhammad naveed</title>
      <link>https://forem.com/muhammad-naveed</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/muhammad-naveed"/>
    <language>en</language>
    <item>
      <title>I Built a Julius AI Alternative in Next.js — Here's What I Learned</title>
      <dc:creator>muhammad naveed</dc:creator>
      <pubDate>Sat, 16 May 2026 11:54:48 +0000</pubDate>
      <link>https://forem.com/muhammad-naveed/i-built-a-julius-ai-alternative-in-nextjs-heres-what-i-learned-2o05</link>
      <guid>https://forem.com/muhammad-naveed/i-built-a-julius-ai-alternative-in-nextjs-heres-what-i-learned-2o05</guid>
      <description>&lt;p&gt;&lt;strong&gt;Tags:&lt;/strong&gt; &lt;code&gt;nextjs&lt;/code&gt; &lt;code&gt;ai&lt;/code&gt; &lt;code&gt;buildinpublic&lt;/code&gt; &lt;code&gt;startup&lt;/code&gt;&lt;/p&gt;




&lt;p&gt;Six weeks ago I started building WhyAnalyst — an AI-powered data analysis tool where you upload a CSV or Excel file and ask questions in plain English. Think Julius AI, but free to start.&lt;/p&gt;

&lt;p&gt;This post is about what actually happened when I built it: the technical decisions, the mistakes, the costs, and the things nobody tells you when you're building an AI SaaS as a solo developer.&lt;/p&gt;




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

&lt;p&gt;Before I get into the lessons, here's what I built it with:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Frontend:  Next.js 14 (App Router)
Auth:      Firebase Authentication
Database:  Firestore
AI:        Google Gemini Flash (switched from GPT-4 — more on this)
Hosting:   Vercel (frontend) + Render (backend API)
Payments:  LemonSqueezy (coming soon)
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Total monthly cost at zero users: &lt;strong&gt;~$0&lt;/strong&gt;. At 100 active free users: still roughly &lt;strong&gt;$0&lt;/strong&gt;. The free tiers on all of these are genuinely generous.&lt;/p&gt;




&lt;h2&gt;
  
  
  The AI cost problem — and how I solved it
&lt;/h2&gt;

&lt;p&gt;This is the thing that almost killed the project before it started.&lt;/p&gt;

&lt;p&gt;My first implementation was naive: user uploads CSV → I send the entire file to the AI → AI answers the question. For a 500-row CSV with 10 columns, that's easily 5,000–10,000 tokens per query. At GPT-4 pricing, that adds up terrifyingly fast.&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;// ❌ What I started with — extremely expensive&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;response&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;openai&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;chat&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;completions&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;create&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;model&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;gpt-4&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;messages&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[{&lt;/span&gt;
    &lt;span class="na"&gt;role&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="s2"&gt;user&lt;/span&gt;&lt;span class="dl"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
    &lt;span class="na"&gt;content&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="s2"&gt;`Here is my data: &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;JSON&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;stringify&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;entireCSV&lt;/span&gt;&lt;span class="p"&gt;)}&lt;/span&gt;&lt;span class="s2"&gt;\n\nQuestion: &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;userQuestion&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="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The fix was to stop sending raw data to the AI entirely. Instead, I send metadata about the data and let the AI generate analysis &lt;em&gt;code&lt;/em&gt;, which runs locally:&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;// ✅ What I do now — much cheaper&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;dataContext&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="na"&gt;columns&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;csvData&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;columns&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;           &lt;span class="c1"&gt;// column names only&lt;/span&gt;
  &lt;span class="na"&gt;sample&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;csvData&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;rows&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;slice&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;5&lt;/span&gt;&lt;span class="p"&gt;),&lt;/span&gt;   &lt;span class="c1"&gt;// first 5 rows only&lt;/span&gt;
  &lt;span class="na"&gt;rowCount&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;csvData&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;rows&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;length&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;      &lt;span class="c1"&gt;// total row count&lt;/span&gt;
  &lt;span class="na"&gt;dtypes&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;inferColumnTypes&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;csvData&lt;/span&gt;&lt;span class="p"&gt;),&lt;/span&gt;  &lt;span class="c1"&gt;// inferred data types&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;response&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;gemini&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;generateContent&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s2"&gt;`
  You are a data analyst. Given this dataset context:
  &lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;JSON&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;stringify&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;dataContext&lt;/span&gt;&lt;span class="p"&gt;)}&lt;/span&gt;&lt;span class="s2"&gt;

  Generate JavaScript code to answer this question: "&lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;userQuestion&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;"
  The full data array is available as the variable 'data'.
  Return only valid JSON: { code: string, chartType: string, title: string }
`&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="c1"&gt;// Execute the generated code against the actual data client-side&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;result&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;Function&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;data&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;generatedCode&lt;/span&gt;&lt;span class="p"&gt;)(&lt;/span&gt;&lt;span class="nx"&gt;csvData&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;rows&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;I also switched from GPT-4 to &lt;strong&gt;Gemini 1.5 Flash&lt;/strong&gt;, which has a generous free tier and is fast enough for this use case. For most CSV analysis questions, the output quality is indistinguishable.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Cost reduction: ~85%&lt;/strong&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  Firebase Auth + Firestore for usage limits
&lt;/h2&gt;

&lt;p&gt;One of the most important things for a freemium AI tool is tracking usage per user so you can enforce limits. Here's the pattern I use:&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;// Called on every analysis attempt&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;checkAndIncrementUsage&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userId&lt;/span&gt;&lt;span class="p"&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;userRef&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;doc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;db&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;users&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;userId&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

  &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;runTransaction&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;db&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="k"&gt;async &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;transaction&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;userDoc&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;transaction&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;get&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userRef&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;queriesUsed&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;queriesLimit&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;plan&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;userDoc&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;data&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;

    &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;queriesUsed&lt;/span&gt; &lt;span class="o"&gt;&amp;gt;=&lt;/span&gt; &lt;span class="nx"&gt;queriesLimit&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
      &lt;span class="k"&gt;throw&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Error&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;LIMIT_REACHED&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="p"&gt;}&lt;/span&gt;

    &lt;span class="nx"&gt;transaction&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;update&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userRef&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
      &lt;span class="na"&gt;queriesUsed&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;increment&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="mi"&gt;1&lt;/span&gt;&lt;span class="p"&gt;),&lt;/span&gt;
      &lt;span class="na"&gt;lastActiveAt&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;serverTimestamp&lt;/span&gt;&lt;span class="p"&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;allowed&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;remaining&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;queriesLimit&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt; &lt;span class="nx"&gt;queriesUsed&lt;/span&gt; &lt;span class="o"&gt;-&lt;/span&gt; &lt;span class="mi"&gt;1&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;I use a Firestore transaction here (not just an update) to avoid race conditions if someone somehow fires two requests simultaneously.&lt;/p&gt;

&lt;p&gt;On signup, I create the user document with defaults:&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;// Firebase Auth onAuthStateChanged → create user doc if new&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;initializeNewUser&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;firebaseUser&lt;/span&gt;&lt;span class="p"&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;userRef&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;doc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;db&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;users&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;firebaseUser&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;uid&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;existing&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;getDoc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userRef&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="o"&gt;!&lt;/span&gt;&lt;span class="nx"&gt;existing&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;exists&lt;/span&gt;&lt;span class="p"&gt;())&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nf"&gt;setDoc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;userRef&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
      &lt;span class="na"&gt;email&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;firebaseUser&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;email&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
      &lt;span class="na"&gt;plan&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;free&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
      &lt;span class="na"&gt;queriesUsed&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="na"&gt;queriesLimit&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mi"&gt;10&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
      &lt;span class="na"&gt;createdAt&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;serverTimestamp&lt;/span&gt;&lt;span class="p"&gt;(),&lt;/span&gt;
      &lt;span class="na"&gt;onboardingComplete&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;false&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;h2&gt;
  
  
  Parsing CSV and Excel on the client
&lt;/h2&gt;

&lt;p&gt;One mistake I made early: sending files to the server for parsing. It's slower, uses server resources, and creates privacy concerns for users with sensitive data. Everything now parses in the browser:&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;Papa&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;papaparse&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="o"&gt;*&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="nx"&gt;XLSX&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;xlsx&lt;/span&gt;&lt;span class="dl"&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;parseFile&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;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;ext&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;file&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="nf"&gt;split&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;.&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;).&lt;/span&gt;&lt;span class="nf"&gt;pop&lt;/span&gt;&lt;span class="p"&gt;().&lt;/span&gt;&lt;span class="nf"&gt;toLowerCase&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;

  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;ext&lt;/span&gt; &lt;span class="o"&gt;===&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;csv&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;return&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Promise&lt;/span&gt;&lt;span class="p"&gt;((&lt;/span&gt;&lt;span class="nx"&gt;resolve&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="nx"&gt;Papa&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;parse&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;span class="p"&gt;{&lt;/span&gt;
        &lt;span class="na"&gt;header&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
        &lt;span class="na"&gt;skipEmptyLines&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
        &lt;span class="na"&gt;complete&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;results&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nf"&gt;resolve&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
          &lt;span class="na"&gt;columns&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;results&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;meta&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;fields&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
          &lt;span class="na"&gt;rows&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nx"&gt;results&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;data&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;span class="p"&gt;}&lt;/span&gt;

  &lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;ext&lt;/span&gt; &lt;span class="o"&gt;===&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;xlsx&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt; &lt;span class="o"&gt;||&lt;/span&gt; &lt;span class="nx"&gt;ext&lt;/span&gt; &lt;span class="o"&gt;===&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;xls&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&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;buffer&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;file&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;arrayBuffer&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;workbook&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;XLSX&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;read&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;buffer&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;sheet&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;workbook&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;Sheets&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="nx"&gt;workbook&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;SheetNames&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="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;rows&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;XLSX&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;utils&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;sheet_to_json&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;sheet&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;columns&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nb"&gt;Object&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;keys&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;rows&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="o"&gt;||&lt;/span&gt; &lt;span class="p"&gt;{}),&lt;/span&gt;
      &lt;span class="nx"&gt;rows&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;This runs instantly even for large files, and the data never leaves the user's browser until they explicitly ask a question.&lt;/p&gt;




&lt;h2&gt;
  
  
  The biggest non-technical mistake I made
&lt;/h2&gt;

&lt;p&gt;I built too many features before talking to any users.&lt;/p&gt;

&lt;p&gt;Look at my sidebar right now: Workspace, Files, Databases, History, Mission Log, Custom Agents, Notebook Templates, Connect Data. Most of these are either empty or barely functional.&lt;/p&gt;

&lt;p&gt;I was building what I imagined users wanted. The reality: every single person who tried the tool just wanted to upload a file and ask a question. That's it. The feature they asked for most often wasn't in any of my sidebar items — it was "can I download the chart as a PNG?"&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Lesson:&lt;/strong&gt; Build the smallest possible thing. Ship it. Watch what real people actually do. Then build the next thing.&lt;/p&gt;




&lt;h2&gt;
  
  
  What's actually working for user acquisition
&lt;/h2&gt;

&lt;p&gt;Since I have zero marketing budget, I've been trying different channels:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Reddit posts&lt;/strong&gt; in r/datascience and r/excel with a demo GIF → best ROI so far&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Building in public on Twitter&lt;/strong&gt; → slow but compounds over time
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;This kind of post&lt;/strong&gt; → you're reading it, so it works at least a little&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;SEO pages&lt;/strong&gt; targeting "julius ai alternative", "chatgpt data analysis alternative" → still building, too early to tell&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;What hasn't worked: posting in Facebook groups, cold DMs, ProductHunt (haven't launched yet, but prep is underway).&lt;/p&gt;




&lt;h2&gt;
  
  
  Current status and what's next
&lt;/h2&gt;

&lt;p&gt;WhyAnalyst is live at &lt;a href="https://whyanalyst.vercel.app" rel="noopener noreferrer"&gt;whyanalyst.vercel.app&lt;/a&gt;. Free tier gives you 10 analyses per month, no credit card required.&lt;/p&gt;

&lt;p&gt;Things I'm working on next:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Chart export (PNG/PDF) — most requested feature&lt;/li&gt;
&lt;li&gt;Persistent file storage so you don't have to re-upload every session&lt;/li&gt;
&lt;li&gt;A Chrome Extension that reads Google Sheets data directly&lt;/li&gt;
&lt;li&gt;Payments via LemonSqueezy for the Pro tier ($9/month)&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Would I do it again?
&lt;/h2&gt;

&lt;p&gt;Yes, but I'd do two things differently:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Talk to 10 potential users before writing a single line of code.&lt;/strong&gt; I would have built a much simpler first version.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Switch to Gemini Flash from day one.&lt;/strong&gt; I wasted time and money on GPT-4 for a use case where Flash is genuinely good enough.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;If you're building something similar — an AI wrapper, a SaaS tool, anything in this space — feel free to ask questions in the comments. Happy to share more about the technical side or the business side.&lt;/p&gt;

&lt;p&gt;And if you want to try the tool (or roast the UI), &lt;a href="https://whyanalyst.vercel.app" rel="noopener noreferrer"&gt;here it is&lt;/a&gt;. Feedback welcome.&lt;/p&gt;




&lt;p&gt;&lt;em&gt;Building WhyAnalyst in public. Follow along if you're into that sort of thing.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>ai</category>
      <category>programming</category>
      <category>automation</category>
      <category>datascience</category>
    </item>
    <item>
      <title>WhyAnalyst: Why Every Data-Driven Decision Needs Better Analysis</title>
      <dc:creator>muhammad naveed</dc:creator>
      <pubDate>Wed, 13 May 2026 07:38:50 +0000</pubDate>
      <link>https://forem.com/muhammad-naveed/whyanalyst-why-every-data-driven-decision-needs-better-analysis-6h1</link>
      <guid>https://forem.com/muhammad-naveed/whyanalyst-why-every-data-driven-decision-needs-better-analysis-6h1</guid>
      <description>&lt;p&gt;Here is the link of app : &lt;a href="https://whyanalyst-438909629186.asia-south1.run.app/" rel="noopener noreferrer"&gt;https://whyanalyst-438909629186.asia-south1.run.app/&lt;/a&gt;&lt;br&gt;
In today’s world, data is everywhere — but insight is rare.&lt;/p&gt;

&lt;p&gt;That’s exactly why I started exploring the idea of WhyAnalyst: a mindset (and potential framework) focused on asking not just what the data says, but why it says it and what to do next.&lt;/p&gt;

&lt;p&gt;📊 The Problem with Modern Data Work&lt;/p&gt;

&lt;p&gt;Most teams today have access to dashboards, BI tools, and analytics platforms. But despite that, a common issue remains:&lt;/p&gt;

&lt;p&gt;Reports are generated, but not deeply understood&lt;br&gt;
Metrics are tracked, but not questioned&lt;br&gt;
Decisions are made on surface-level interpretation&lt;/p&gt;

&lt;p&gt;In short: we have more data, but not always better analysis.&lt;/p&gt;

&lt;p&gt;🧠 What “WhyAnalyst” Means&lt;/p&gt;

&lt;p&gt;WhyAnalyst is about shifting from passive reporting to active reasoning.&lt;/p&gt;

&lt;p&gt;Instead of stopping at:&lt;/p&gt;

&lt;p&gt;“Sales dropped 12%”&lt;/p&gt;

&lt;p&gt;We go further:&lt;/p&gt;

&lt;p&gt;Why did sales drop?&lt;br&gt;
Was it traffic, conversion, pricing, or seasonality?&lt;br&gt;
Which segment was most affected?&lt;br&gt;
What changed before the drop occurred?&lt;/p&gt;

&lt;p&gt;It’s about building a habit of analytical curiosity.&lt;/p&gt;

&lt;p&gt;🔍 Core Principles of WhyAnalyst&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Always Ask “Why?”&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Never accept surface-level metrics without investigation.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Break Down Every Metric&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Every number is a combination of smaller drivers.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Look for Causality, Not Just Correlation&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Not everything that moves together is connected.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Focus on Actionable Insight&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;Analysis is only useful if it leads to decisions.&lt;/p&gt;

&lt;p&gt;⚙️ Where This Applies&lt;/p&gt;

&lt;p&gt;WhyAnalyst thinking is useful in:&lt;/p&gt;

&lt;p&gt;Data analytics &amp;amp; business intelligence&lt;br&gt;
Product analytics&lt;br&gt;
Marketing performance analysis&lt;br&gt;
AI model evaluation&lt;br&gt;
Financial decision-making&lt;/p&gt;

&lt;p&gt;Anywhere data is used — this mindset improves outcomes.&lt;/p&gt;

&lt;p&gt;🚀 Why It Matters&lt;/p&gt;

&lt;p&gt;As AI and automation grow, raw analysis is becoming cheaper.&lt;/p&gt;

&lt;p&gt;What remains valuable is:&lt;/p&gt;

&lt;p&gt;Critical thinking&lt;br&gt;
Interpretation&lt;br&gt;
Decision-making based on uncertainty&lt;/p&gt;

&lt;p&gt;That’s where human analysts still win — and where WhyAnalyst fits in.&lt;/p&gt;

&lt;p&gt;🤝 Final Thought&lt;/p&gt;

&lt;p&gt;WhyAnalyst is not a tool — it’s a habit.&lt;/p&gt;

&lt;p&gt;A habit of refusing to stop at “what happened” and always pushing toward “why it happened” and “what should happen next.”&lt;/p&gt;

</description>
      <category>whyanalyst</category>
      <category>datascience</category>
      <category>ai</category>
      <category>googlecloud</category>
    </item>
    <item>
      <title>My Journey to Becoming a Full-Stack Developer</title>
      <dc:creator>muhammad naveed</dc:creator>
      <pubDate>Tue, 18 Mar 2025 06:24:50 +0000</pubDate>
      <link>https://forem.com/muhammad-naveed/my-journey-to-becoming-a-full-stack-developer-1c6j</link>
      <guid>https://forem.com/muhammad-naveed/my-journey-to-becoming-a-full-stack-developer-1c6j</guid>
      <description>&lt;p&gt;Introduction&lt;/p&gt;

&lt;p&gt;Hello, Dev.to community! My name is Muhammad Naveed, and I am on an exciting journey to becoming a full-stack developer. I started my journey with basic web development and am now diving deep into frontend and backend technologies. In this post, I’ll share my experience, challenges, and the roadmap that helped me get started.&lt;/p&gt;

&lt;p&gt;Getting Started with Web Development&lt;/p&gt;

&lt;p&gt;Like many beginners, I started with the fundamentals:&lt;/p&gt;

&lt;p&gt;HTML &amp;amp; CSS: Learning the structure and styling of web pages.&lt;/p&gt;

&lt;p&gt;Bootstrap: To quickly create responsive designs.&lt;/p&gt;

&lt;p&gt;JavaScript: To make my websites interactive.&lt;/p&gt;

&lt;p&gt;It took me about a month to feel comfortable with these technologies, and I built a few basic projects like landing pages and portfolio websites.&lt;/p&gt;

&lt;p&gt;Diving into JavaScript and DOM Manipulation&lt;/p&gt;

&lt;p&gt;After mastering the basics, I focused on JavaScript:&lt;/p&gt;

&lt;p&gt;DOM Manipulation: Understanding how to update the UI dynamically.&lt;/p&gt;

&lt;p&gt;Functions &amp;amp; Callbacks: Learning how to structure JavaScript code.&lt;/p&gt;

&lt;p&gt;APIs: Exploring how to fetch data using the Unsplash API.&lt;/p&gt;

&lt;p&gt;One of the most important things I learned was how JavaScript handles asynchronous operations, which prepared me for working with APIs and backend technologies.&lt;/p&gt;

&lt;p&gt;Moving Towards Full-Stack Development&lt;/p&gt;

&lt;p&gt;Once I gained confidence in frontend development, I started learning backend technologies:&lt;/p&gt;

&lt;p&gt;Node.js &amp;amp; Express.js: To build server-side applications.&lt;/p&gt;

&lt;p&gt;MongoDB: To store and manage data.&lt;/p&gt;

&lt;p&gt;REST APIs: Creating and consuming APIs.&lt;/p&gt;

&lt;p&gt;I also started learning about authentication, authorization, and deploying applications on cloud platforms.&lt;/p&gt;

&lt;p&gt;Challenges I Faced&lt;/p&gt;

&lt;p&gt;Like any learning journey, I faced challenges such as:&lt;/p&gt;

&lt;p&gt;Understanding complex JavaScript concepts like closures and promises.&lt;/p&gt;

&lt;p&gt;Debugging errors and learning how to use browser dev tools effectively.&lt;/p&gt;

&lt;p&gt;Managing time effectively while learning full-time.&lt;/p&gt;

&lt;p&gt;The key to overcoming these challenges was consistency, building projects, and engaging with the developer community.&lt;/p&gt;

&lt;p&gt;My Next Steps&lt;/p&gt;

&lt;p&gt;As I continue my learning, I plan to:&lt;/p&gt;

&lt;p&gt;Work on more full-stack projects.&lt;/p&gt;

&lt;p&gt;Explore React.js and Angular for frontend frameworks.&lt;/p&gt;

&lt;p&gt;Improve my problem-solving skills by practicing algorithms and data structures.&lt;/p&gt;

&lt;p&gt;Conclusion&lt;/p&gt;

&lt;p&gt;The journey to becoming a full-stack developer is challenging but rewarding. If you're starting, my advice is: stay consistent, build projects, and never stop learning!&lt;/p&gt;

&lt;p&gt;What’s your full-stack development journey like? Let’s discuss in the comments!&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>frontend</category>
      <category>backend</category>
      <category>react</category>
    </item>
  </channel>
</rss>
