<?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: Mahesh Srinivasan</title>
    <description>The latest articles on DEV Community by Mahesh Srinivasan (@mahesh-dev).</description>
    <link>https://dev.to/mahesh-dev</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%2F3887136%2F63ed56f0-ba78-4377-a48a-42c6d570e622.jpg</url>
      <title>DEV Community: Mahesh Srinivasan</title>
      <link>https://dev.to/mahesh-dev</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/mahesh-dev"/>
    <language>en</language>
    <item>
      <title>[Boost]</title>
      <dc:creator>Mahesh Srinivasan</dc:creator>
      <pubDate>Mon, 15 Jun 2026 04:25:56 +0000</pubDate>
      <link>https://dev.to/mahesh-dev/-5c4p</link>
      <guid>https://dev.to/mahesh-dev/-5c4p</guid>
      <description>&lt;div class="ltag__link--embedded"&gt;
  &lt;div class="crayons-story "&gt;
  &lt;a href="https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak" class="crayons-story__hidden-navigation-link"&gt;Stop Using ngrok for Webhook Testing (There's a Much Better Way Now)&lt;/a&gt;


  &lt;div class="crayons-story__body crayons-story__body-full_post"&gt;
    &lt;div class="crayons-story__top"&gt;
      &lt;div class="crayons-story__meta"&gt;
        &lt;div class="crayons-story__author-pic"&gt;

          &lt;a href="/mahesh-dev" class="crayons-avatar  crayons-avatar--l  "&gt;
            &lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3887136%2F63ed56f0-ba78-4377-a48a-42c6d570e622.jpg" alt="mahesh-dev profile" class="crayons-avatar__image" width="350" height="350"&gt;
          &lt;/a&gt;
        &lt;/div&gt;
        &lt;div&gt;
          &lt;div&gt;
            &lt;a href="/mahesh-dev" class="crayons-story__secondary fw-medium m:hidden"&gt;
              Mahesh Srinivasan
            &lt;/a&gt;
            &lt;div class="profile-preview-card relative mb-4 s:mb-0 fw-medium hidden m:inline-block"&gt;
              
                Mahesh Srinivasan
                
              
              &lt;div id="story-author-preview-content-3522077" class="profile-preview-card__content crayons-dropdown branded-7 p-4 pt-0"&gt;
                &lt;div class="gap-4 grid"&gt;
                  &lt;div class="-mt-4"&gt;
                    &lt;a href="/mahesh-dev" class="flex"&gt;
                      &lt;span class="crayons-avatar crayons-avatar--xl mr-2 shrink-0"&gt;
                        &lt;img src="https://media2.dev.to/dynamic/image/width=800%2Cheight=%2Cfit=scale-down%2Cgravity=auto%2Cformat=auto/https%3A%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3887136%2F63ed56f0-ba78-4377-a48a-42c6d570e622.jpg" class="crayons-avatar__image" alt="" width="350" height="350"&gt;
                      &lt;/span&gt;
                      &lt;span class="crayons-link crayons-subtitle-2 mt-5"&gt;Mahesh Srinivasan&lt;/span&gt;
                    &lt;/a&gt;
                  &lt;/div&gt;
                  &lt;div class="print-hidden"&gt;
                    
                      Follow
                    
                  &lt;/div&gt;
                  &lt;div class="author-preview-metadata-container"&gt;&lt;/div&gt;
                &lt;/div&gt;
              &lt;/div&gt;
            &lt;/div&gt;

          &lt;/div&gt;
          &lt;a href="https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak" class="crayons-story__tertiary fs-xs"&gt;&lt;time&gt;Apr 19&lt;/time&gt;&lt;span class="time-ago-indicator-initial-placeholder"&gt;&lt;/span&gt;&lt;/a&gt;
        &lt;/div&gt;
      &lt;/div&gt;

    &lt;/div&gt;

    &lt;div class="crayons-story__indention"&gt;
      &lt;h2 class="crayons-story__title crayons-story__title-full_post"&gt;
        &lt;a href="https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak" id="article-link-3522077"&gt;
          Stop Using ngrok for Webhook Testing (There's a Much Better Way Now)
        &lt;/a&gt;
      &lt;/h2&gt;
        &lt;div class="crayons-story__tags"&gt;
            &lt;a class="crayons-tag  crayons-tag--monochrome " href="/t/api"&gt;&lt;span class="crayons-tag__prefix"&gt;#&lt;/span&gt;api&lt;/a&gt;
            &lt;a class="crayons-tag  crayons-tag--monochrome " href="/t/productivity"&gt;&lt;span class="crayons-tag__prefix"&gt;#&lt;/span&gt;productivity&lt;/a&gt;
            &lt;a class="crayons-tag  crayons-tag--monochrome " href="/t/testing"&gt;&lt;span class="crayons-tag__prefix"&gt;#&lt;/span&gt;testing&lt;/a&gt;
            &lt;a class="crayons-tag  crayons-tag--monochrome " href="/t/tooling"&gt;&lt;span class="crayons-tag__prefix"&gt;#&lt;/span&gt;tooling&lt;/a&gt;
        &lt;/div&gt;
      &lt;div class="crayons-story__bottom"&gt;
        &lt;div class="crayons-story__details"&gt;
            &lt;a href="https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak#comments" class="crayons-btn crayons-btn--s crayons-btn--ghost crayons-btn--icon-left flex items-center"&gt;
              

              &lt;span class="hidden s:inline"&gt;Add&amp;nbsp;Comment&lt;/span&gt;
            &lt;/a&gt;
        &lt;/div&gt;
        &lt;div class="crayons-story__save"&gt;
          &lt;small class="crayons-story__tertiary fs-xs mr-2"&gt;
            4 min read
          &lt;/small&gt;
            
              &lt;span class="bm-initial crayons-icon c-btn__icon"&gt;
                

              &lt;/span&gt;
              &lt;span class="bm-success crayons-icon c-btn__icon"&gt;
                

              &lt;/span&gt;
            
        &lt;/div&gt;
      &lt;/div&gt;
    &lt;/div&gt;
  &lt;/div&gt;
&lt;/div&gt;

&lt;/div&gt;


</description>
    </item>
    <item>
      <title>Stop Using ngrok for Webhook Testing (There's a Much Better Way Now)</title>
      <dc:creator>Mahesh Srinivasan</dc:creator>
      <pubDate>Sun, 19 Apr 2026 09:19:48 +0000</pubDate>
      <link>https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak</link>
      <guid>https://dev.to/mahesh-dev/stop-using-ngrok-for-webhook-testing-a-simpler-way-37ak</guid>
      <description>&lt;p&gt;Debugging webhooks has always been… annoying.&lt;/p&gt;

&lt;p&gt;If you've worked with Stripe, GitHub, or any webhook-based system, you probably know this flow:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Start your local server&lt;/li&gt;
&lt;li&gt;Run ngrok&lt;/li&gt;
&lt;li&gt;Copy the public URL&lt;/li&gt;
&lt;li&gt;Paste it into your provider&lt;/li&gt;
&lt;li&gt;Restart ngrok → URL changes&lt;/li&gt;
&lt;li&gt;Repeat everything again&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;It works… but it's not smooth.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Real Problems
&lt;/h2&gt;

&lt;p&gt;After using this workflow for a while, I realized ngrok isn't just inconvenient — it has fundamental gaps:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;URLs change on every restart&lt;/strong&gt; — you're constantly updating provider dashboards&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No replay&lt;/strong&gt; — if you missed a request, it's gone&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No visibility into what your server returned&lt;/strong&gt; — you're debugging blind&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Signature verification is manual&lt;/strong&gt; — you write the HMAC check yourself every time&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No history&lt;/strong&gt; — you can't go back and look at past requests&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;Tunnel management is a whole side job&lt;/strong&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  What I Wanted Instead
&lt;/h2&gt;

&lt;p&gt;I wanted one tool that:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Gives me a stable, permanent URL for each provider I test against&lt;/li&gt;
&lt;li&gt;Streams requests to my terminal in real-time&lt;/li&gt;
&lt;li&gt;Forwards them to localhost automatically&lt;/li&gt;
&lt;li&gt;Shows me what my server actually returned&lt;/li&gt;
&lt;li&gt;Lets me replay missed requests with one command&lt;/li&gt;
&lt;li&gt;Handles signature verification so I don't have to&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  So I Built Anonymily
&lt;/h2&gt;

&lt;p&gt;A webhook development platform — not just a tunnel.&lt;/p&gt;

&lt;h2&gt;
  
  
  How It Works
&lt;/h2&gt;

&lt;p&gt;Start your local server, then run:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @anonymilyhq/cli listen 3000
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;You instantly get:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;Forwarding to http://localhost:3000
Webhook URL: https://api.anonymily.com/h/abc12345
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Send your webhook to that URL. Anonymily captures it, streams it to your CLI in real-time, forwards it to &lt;code&gt;localhost:3000&lt;/code&gt;, and captures your server's response — all in one shot.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight http"&gt;&lt;code&gt;&lt;span class="err"&gt;POST /webhook 200 OK (120ms)
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  What's Different From ngrok
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Stable Named Endpoints (Pro)
&lt;/h3&gt;

&lt;p&gt;This is the biggest quality-of-life upgrade. Instead of a random URL that changes every session, you can claim a permanent slug:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @anonymilyhq/cli listen 3000 &lt;span class="nt"&gt;--id&lt;/span&gt; stripe-dev &lt;span class="nt"&gt;--token&lt;/span&gt; &amp;lt;PAT&amp;gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Your webhook URL is now &lt;code&gt;https://api.anonymily.com/h/stripe-dev&lt;/code&gt;. Set it in your Stripe dashboard once and never touch it again — even if you restart the CLI, redeploy, or switch machines.&lt;/p&gt;

&lt;h3&gt;
  
  
  Replay Any Request
&lt;/h3&gt;

&lt;p&gt;Missed a webhook while your server was down? Replay it:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @anonymilyhq/cli replay &amp;lt;hookId&amp;gt; &amp;lt;requestId&amp;gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The CLI fetches the original request body, method, and headers, then re-sends it to your localhost. You can override the method or body inline, or re-sign the payload with your signing secret.&lt;/p&gt;

&lt;h3&gt;
  
  
  Response Capture
&lt;/h3&gt;

&lt;p&gt;The CLI automatically captures what your local server returns and streams it back to the dashboard. You can see status code, latency, headers, and body — without touching your server logs.&lt;/p&gt;

&lt;h3&gt;
  
  
  Signature Verification Built In
&lt;/h3&gt;

&lt;p&gt;Anonymily auto-detects and verifies HMAC signatures for Stripe, GitHub, Shopify, Slack, and Razorpay. You see immediately whether a request is valid or forged — before it even hits your code.&lt;/p&gt;

&lt;h3&gt;
  
  
  Real-Time Dashboard
&lt;/h3&gt;

&lt;p&gt;Beyond the CLI, there's a full web dashboard at &lt;code&gt;anonymily.com&lt;/code&gt;. Every request shows up instantly with headers, body, the signature verification result, your server's response, and a one-click replay button.&lt;/p&gt;

&lt;h3&gt;
  
  
  Search and Filter
&lt;/h3&gt;

&lt;p&gt;When you've got hundreds of test requests piling up, you can filter by method, status code, or search across headers and body to find exactly what you need.&lt;/p&gt;

&lt;h2&gt;
  
  
  AI Features (Pro)
&lt;/h2&gt;

&lt;p&gt;This is where it gets interesting for debugging.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;AI Diagnosis&lt;/strong&gt; — When something goes wrong (signature mismatch, 500 from your server, unexpected payload shape), you can ask for an AI diagnosis. It reads the full exchange — request, signature result, your server's response — and explains exactly what went wrong and how to fix it.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;AI Handler Generation&lt;/strong&gt; — Point it at a captured request and it generates a production-ready handler in Express, Next.js, or NestJS, grounded in your actual payload shape. Not a generic template — code that handles your specific webhook structure.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Synthetic Event Triggers&lt;/strong&gt; — Fire realistic test events for any supported provider without needing the real service to send them:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @anonymilyhq/cli trigger stripe payment_intent.succeeded &lt;span class="nt"&gt;--hook&lt;/span&gt; stripe-dev &lt;span class="nt"&gt;--token&lt;/span&gt; &amp;lt;PAT&amp;gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Events are correctly signed with your hook's signing secret.&lt;/p&gt;

&lt;h2&gt;
  
  
  Use It From Claude Code (MCP)
&lt;/h2&gt;

&lt;p&gt;If you use Claude Code, you can connect Anonymily as an MCP server and call all these features directly from your AI assistant:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"mcpServers"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"anonymily"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"command"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"npx"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"args"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"@anonymilyhq/mcp-server"&lt;/span&gt;&lt;span class="p"&gt;],&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="nl"&gt;"env"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"ANONYMILY_TOKEN"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"&amp;lt;your-PAT&amp;gt;"&lt;/span&gt;&lt;span class="w"&gt;
      &lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;From there, Claude can list your captured requests, replay them, trigger synthetic events, and generate handler code — all without leaving your editor.&lt;/p&gt;

&lt;h2&gt;
  
  
  Pricing
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Free&lt;/strong&gt; — No account needed. Get a random endpoint instantly. 200 requests stored, 48-hour history, 5 replays per day.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Pro ($9/month)&lt;/strong&gt; — Custom named endpoints, 2,000 requests stored, 30-day history, unlimited replays, signature verification, AI features, signing secrets.&lt;/p&gt;

&lt;p&gt;The free tier requires zero signup. If you just want a quick endpoint to test a webhook right now, run the &lt;code&gt;npx&lt;/code&gt; command and you're done.&lt;/p&gt;

&lt;h2&gt;
  
  
  Works With Anything
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Stripe webhooks&lt;/li&gt;
&lt;li&gt;GitHub events&lt;/li&gt;
&lt;li&gt;Shopify hooks&lt;/li&gt;
&lt;li&gt;Slack events&lt;/li&gt;
&lt;li&gt;Razorpay payment events&lt;/li&gt;
&lt;li&gt;Any custom HTTP webhook&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;If it sends HTTP requests, it works.&lt;/p&gt;

&lt;h2&gt;
  
  
  Try It
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npx @anonymilyhq/cli listen 3000
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's it. No install, no account, no config.&lt;/p&gt;

&lt;p&gt;If you're building something that involves webhooks — local dev, CI testing, provider integration — give it a try:&lt;/p&gt;

&lt;p&gt;👉 &lt;a href="https://anonymily.com" rel="noopener noreferrer"&gt;https://anonymily.com&lt;/a&gt;&lt;/p&gt;

&lt;h2&gt;
  
  
  Feedback Welcome
&lt;/h2&gt;

&lt;p&gt;This is still early, and I'm actively iterating. Would love to hear:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;What's missing for your workflow?&lt;/li&gt;
&lt;li&gt;What providers would you want synthetic events for?&lt;/li&gt;
&lt;li&gt;Any friction in the CLI or dashboard?&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Drop thoughts below or open an issue. 👇&lt;/p&gt;

</description>
      <category>api</category>
      <category>productivity</category>
      <category>testing</category>
      <category>tooling</category>
    </item>
  </channel>
</rss>
