<?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: Tobiloba</title>
    <description>The latest articles on DEV Community by Tobiloba (@towbee98).</description>
    <link>https://dev.to/towbee98</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%2F476907%2Fabbe5f5d-5dca-4e0d-9f5f-60b42f2288ed.jpeg</url>
      <title>DEV Community: Tobiloba</title>
      <link>https://dev.to/towbee98</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/towbee98"/>
    <language>en</language>
    <item>
      <title>How I Built a Dynamic Profile API with Live Cat Facts</title>
      <dc:creator>Tobiloba</dc:creator>
      <pubDate>Mon, 20 Oct 2025 00:59:55 +0000</pubDate>
      <link>https://dev.to/towbee98/how-i-built-a-dynamic-profile-api-with-live-cat-facts-12p5</link>
      <guid>https://dev.to/towbee98/how-i-built-a-dynamic-profile-api-with-live-cat-facts-12p5</guid>
      <description>&lt;p&gt;When I saw the HNG Backend Stage 0 challenge build a /me endpoint that returns my profile plus a live cat fact,I thought: “Simple, right?”&lt;br&gt;
Turns out, even “simple” APIs teach you big lessons about error handling, module systems, and the chaos of third-party dependencies. Here’s how I built it, what went wrong, and how I fixed it all in TypeScript + Express, deployed from my local machine.&lt;/p&gt;

&lt;h2&gt;
  
  
  🎯 Why This Task Matters
&lt;/h2&gt;

&lt;p&gt;This isn’t just a JSON blob. It’s a microcosm of real backend work:&lt;/p&gt;

&lt;p&gt;✅ Dynamic data: Fresh timestamp on every request&lt;br&gt;
🌐 External API integration: Real-time cat facts from catfact.ninja&lt;br&gt;
🛡️ Fault tolerance: Graceful fallbacks when APIs fail&lt;br&gt;
🚪 Instant public access: No cloud deployment needed thanks to ngrok!&lt;/p&gt;

&lt;h2&gt;
  
  
  The Core Logic: /me Handler
&lt;/h2&gt;

&lt;p&gt;The magic happens in under 30 lines. Key features:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Live timestamp (ISO 8601 UTC)&lt;/li&gt;
&lt;li&gt;Fresh cat fact per request (with 5s timeout)&lt;/li&gt;
&lt;li&gt;Fallback fact if the cat API flakes out&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2Fg3z8ouv4xfo5wr7r12gc.png" class="article-body-image-wrapper"&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%2Farticles%2Fg3z8ouv4xfo5wr7r12gc.png" alt="The code" width="800" height="875"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h2&gt;
  
  
  🚧 The Module System Trap (ESM vs CommonJS)
&lt;/h2&gt;

&lt;p&gt;My biggest headache? This runtime crash:&lt;/p&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2Fzhwb4jyigm2gy7kee6vw.png" class="article-body-image-wrapper"&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%2Farticles%2Fzhwb4jyigm2gy7kee6vw.png" alt="The error i faced" width="800" height="282"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Cause: I used ESM-style named imports with a CommonJS package (express):&lt;/p&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2F1jlcj5qpc8jvio5vmupy.png" class="article-body-image-wrapper"&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%2Farticles%2F1jlcj5qpc8jvio5vmupy.png" alt="What caused it" width="800" height="285"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Fix: Use type-only aliases (zero runtime cost):&lt;/p&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2Ffiimm0tku2ug2oxk3r1u.png" class="article-body-image-wrapper"&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%2Farticles%2Ffiimm0tku2ug2oxk3r1u.png" alt="The solution" width="800" height="471"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Lesson: TypeScript compiles ≠ Node.js runs. Always test with tsx or compiled JS!&lt;/p&gt;

&lt;h2&gt;
  
  
  🌐 Deployment? Meet ngrok.
&lt;/h2&gt;

&lt;p&gt;Since cloud platforms like Vercel were off-limits, I went local first:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Ran my Express server: npm run dev → &lt;a href="http://localhost:3000" rel="noopener noreferrer"&gt;http://localhost:3000&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;Fired up ngrok: &lt;code&gt;ngrok http 3000&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;Got an instant public URL: &lt;a href="https://f8ffe71c697b.ngrok-free.app" rel="noopener noreferrer"&gt;https://f8ffe71c697b.ngrok-free.app&lt;/a&gt;
&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;✅ No Docker&lt;br&gt;
✅ No CI/CD&lt;br&gt;
✅ No waiting for builds&lt;/p&gt;

&lt;p&gt;Just pure, tunnelled, localhost magic.&lt;/p&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2F0nr62ckjwqu8715nn2sk.png" class="article-body-image-wrapper"&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%2Farticles%2F0nr62ckjwqu8715nn2sk.png" alt="Ngrok terminal" width="800" height="450"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="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%2Farticles%2Fqno1dfv6mwo20rygzcoe.png" class="article-body-image-wrapper"&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%2Farticles%2Fqno1dfv6mwo20rygzcoe.png" alt="the url in the browser" width="800" height="450"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;h2&gt;
  
  
  🔑 Key Takeaways
&lt;/h2&gt;

&lt;ol&gt;
&lt;li&gt;Local can be public
ngrok turns localhost into a shareable URL in seconds—perfect  for demos, testing, and challenges like this.&lt;/li&gt;
&lt;li&gt;Third-party APIs will fail
Always code for failure. A fallback fact kept my endpoint alive during catfact.ninja outages.&lt;/li&gt;
&lt;li&gt;ESM + CommonJS = handle with care
Use type aliases for Express types in ESM projects. Avoid runtime destructuring.&lt;/li&gt;
&lt;li&gt;Small tasks, big insights
This “simple” endpoint taught me about timeouts, env vars, module systems, and resilient design.&lt;/li&gt;
&lt;/ol&gt;

&lt;h2&gt;
  
  
  📦 Try It Yourself!
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Github Repo: &lt;a href="https://github.com/towbee98/fictional-octo-chainsaw" rel="noopener noreferrer"&gt;https://github.com/towbee98/fictional-octo-chainsaw&lt;/a&gt;
&lt;/li&gt;
&lt;li&gt;Run locally:
&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%2Farticles%2Fw1ybrren3289po8hzgpb.png" alt="How to run locally" width="800" height="472"&gt;
Then hit your ngrok URL + /me and watch the cat facts roll in! 🐾&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  👋 Final Thought
&lt;/h2&gt;

&lt;p&gt;You don’t need a server farm to build something useful. With TypeScript, Express, and ngrok, your laptop becomes a global API endpoint.&lt;/p&gt;

&lt;p&gt;And if nothing else,now you know that cats spend 70% of their lives sleeping. Go take a nap. You’ve earned it.&lt;/p&gt;

&lt;p&gt;Built for Backend Wizards . Stage 0: complete.&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>programming</category>
      <category>backend</category>
      <category>typescript</category>
    </item>
  </channel>
</rss>
