<?xml version="1.0" encoding="utf-8"?><?xml-stylesheet type="text/xsl" href="atom.xsl"?>
<feed xmlns="http://www.w3.org/2005/Atom">
    <id>https://hostim.dev/blog/</id>
    <title>HOSTIM.DEV Blog</title>
    <updated>2026-04-09T00:00:00.000Z</updated>
    <generator>https://github.com/jpmonette/feed</generator>
    <link rel="alternate" href="https://hostim.dev/blog/"/>
    <subtitle>HOSTIM.DEV Blog</subtitle>
    <icon>https://hostim.dev/img/favicon.ico</icon>
    <entry>
        <title type="html"><![CDATA[Which Database Should You Self-Host? SQLite vs MySQL vs PostgreSQL vs Redis]]></title>
        <id>https://hostim.dev/blog/database-showdown/</id>
        <link href="https://hostim.dev/blog/database-showdown/"/>
        <updated>2026-04-09T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[SQLite, MySQL, PostgreSQL, and Redis each solve different problems. Here's how they compare for self-hosted apps – with pros, cons, and when to use each.]]></summary>
        <content type="html"><![CDATA[<p>When you're deploying your own app, the database choice matters more than most people think. It affects performance, ops complexity, backups, and how much memory your server needs.</p>
<p>There are four options you'll run into most often: <strong>SQLite, MySQL, PostgreSQL, and Redis</strong>. They're not all the same kind of database – and that's the point. Here's when each one makes sense.</p>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="sqlite--the-zero-ops-embedded-database">SQLite – the zero-ops embedded database<a href="https://hostim.dev/blog/database-showdown/#sqlite--the-zero-ops-embedded-database" class="hash-link" aria-label="Direct link to SQLite – the zero-ops embedded database" title="Direct link to SQLite – the zero-ops embedded database" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> small apps, prototypes, CLIs, single-user tools, edge deployments</li>
<li class=""><strong>Strengths:</strong> no server process, single file, zero config, instant setup</li>
<li class=""><strong>Weaknesses:</strong> no concurrent writes, no replication, hard to scale past one instance</li>
</ul>
<p>SQLite is not a server – it's a library that reads and writes a single file. That makes it perfect for apps where simplicity matters more than scale. If your app has one process writing to the database and modest traffic, SQLite will outperform anything else because there's no network round-trip. The moment you need concurrent writes or multiple app replicas, you've outgrown it.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="mysql--the-reliable-workhorse">MySQL – the reliable workhorse<a href="https://hostim.dev/blog/database-showdown/#mysql--the-reliable-workhorse" class="hash-link" aria-label="Direct link to MySQL – the reliable workhorse" title="Direct link to MySQL – the reliable workhorse" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> web apps, CMS platforms, CRUD-heavy workloads, WordPress/Laravel stacks</li>
<li class=""><strong>Strengths:</strong> fast reads, mature replication, huge ecosystem, low memory footprint</li>
<li class=""><strong>Weaknesses:</strong> weaker JSON support, less strict by default, fewer advanced types</li>
</ul>
<p>MySQL powers a massive chunk of the internet. It's battle-tested, well-documented, and runs well even on small VPS instances. If you're running a standard web app with mostly reads and simple queries, MySQL will serve you well without hogging resources. Just be aware that its default configs are more lenient than PostgreSQL – silent truncations and implicit type casts can bite you.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="postgresql--the-feature-rich-powerhouse">PostgreSQL – the feature-rich powerhouse<a href="https://hostim.dev/blog/database-showdown/#postgresql--the-feature-rich-powerhouse" class="hash-link" aria-label="Direct link to PostgreSQL – the feature-rich powerhouse" title="Direct link to PostgreSQL – the feature-rich powerhouse" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> complex queries, data integrity, JSON workloads, GIS, analytics</li>
<li class=""><strong>Strengths:</strong> advanced types (JSONB, arrays, hstore), strong standards compliance, extensions ecosystem</li>
<li class=""><strong>Weaknesses:</strong> higher memory usage, more tuning needed, steeper learning curve for ops</li>
</ul>
<p>PostgreSQL is the database you pick when correctness and flexibility matter. It handles complex joins, window functions, CTEs, and full-text search natively. The extension ecosystem (PostGIS, pg_cron, pgvector) makes it a Swiss army knife. The trade-off: it's hungrier on resources and rewards careful tuning of <code>shared_buffers</code>, <code>work_mem</code>, and connection pooling.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="redis--the-in-memory-speed-layer">Redis – the in-memory speed layer<a href="https://hostim.dev/blog/database-showdown/#redis--the-in-memory-speed-layer" class="hash-link" aria-label="Direct link to Redis – the in-memory speed layer" title="Direct link to Redis – the in-memory speed layer" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> caching, sessions, rate limiting, queues, pub/sub, leaderboards</li>
<li class=""><strong>Strengths:</strong> sub-millisecond reads, rich data structures (lists, sets, sorted sets, streams), built-in TTL</li>
<li class=""><strong>Weaknesses:</strong> data must fit in RAM, persistence is optional and lossy, not a primary data store</li>
</ul>
<p>Redis isn't a replacement for a relational database – it's a complement. Use it for things that need to be fast and can tolerate occasional data loss: session tokens, cache layers, job queues. Redis Streams can even replace simple message brokers. Just don't store your source of truth here – if the server restarts between RDB snapshots, recent writes are gone.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="quick-comparison">Quick comparison<a href="https://hostim.dev/blog/database-showdown/#quick-comparison" class="hash-link" aria-label="Direct link to Quick comparison" title="Direct link to Quick comparison" translate="no">​</a></h2>
<table><thead><tr><th>Feature</th><th><strong>SQLite</strong></th><th><strong>MySQL</strong></th><th><strong>PostgreSQL</strong></th><th><strong>Redis</strong></th></tr></thead><tbody><tr><td>Type</td><td>Embedded</td><td>Relational server</td><td>Relational server</td><td>In-memory store</td></tr><tr><td>Ease of setup</td><td>⭐⭐⭐⭐⭐</td><td>⭐⭐⭐⭐</td><td>⭐⭐⭐</td><td>⭐⭐⭐⭐</td></tr><tr><td>Concurrent writes</td><td>❌ Single-writer</td><td>✅ Good</td><td>✅ Excellent</td><td>✅ Very fast</td></tr><tr><td>Complex queries</td><td>Basic</td><td>Good</td><td>Excellent</td><td>N/A (key-value)</td></tr><tr><td>Memory usage</td><td>Minimal</td><td>Low–moderate</td><td>Moderate–high</td><td>High (all data in RAM)</td></tr><tr><td>Replication</td><td>None built-in</td><td>Mature</td><td>Mature</td><td>Built-in</td></tr><tr><td>Best self-host size</td><td>Single instance</td><td>Small–large</td><td>Medium–large</td><td>Any (as cache layer)</td></tr><tr><td>Persistence</td><td>Always (file)</td><td>Always (disk)</td><td>Always (disk)</td><td>Optional (RDB/AOF)</td></tr></tbody></table>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="so-which-one-should-you-choose">So which one should you choose?<a href="https://hostim.dev/blog/database-showdown/#so-which-one-should-you-choose" class="hash-link" aria-label="Direct link to So which one should you choose?" title="Direct link to So which one should you choose?" translate="no">​</a></h2>
<ul>
<li class=""><strong>Building a prototype or CLI tool?</strong> → <strong>SQLite</strong></li>
<li class=""><strong>Running a standard web app?</strong> → <strong>MySQL</strong></li>
<li class=""><strong>Need complex queries, JSONB, or extensions?</strong> → <strong>PostgreSQL</strong></li>
<li class=""><strong>Need a fast cache, session store, or queue?</strong> → <strong>Redis</strong></li>
</ul>
<p>Most real-world apps end up using <strong>two</strong>: a relational database (MySQL or PostgreSQL) for your data, and Redis for caching and sessions. That's not overkill – it's the right tool for each job.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="self-hosting-these-databases">Self-hosting these databases<a href="https://hostim.dev/blog/database-showdown/#self-hosting-these-databases" class="hash-link" aria-label="Direct link to Self-hosting these databases" title="Direct link to Self-hosting these databases" translate="no">​</a></h2>
<p>Running databases on a VPS means managing backups, updates, and disk space yourself. It's doable, but it's one more thing to maintain.</p>
<p>On <a href="https://hostim.dev/" target="_blank" rel="noopener noreferrer" class="">Hostim.dev</a>, MySQL, PostgreSQL, and Redis are built in – provisioned alongside your app with metrics and no extra config. Paste a <code>docker-compose.yml</code> and your database is ready.</p>
<p>👉 <a href="https://console.hostim.dev/" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console"><b>Try it free</b></a></p>]]></content>
        <category label="database" term="database"/>
        <category label="sqlite" term="sqlite"/>
        <category label="mysql" term="mysql"/>
        <category label="postgresql" term="postgresql"/>
        <category label="redis" term="redis"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Heroku Is Freezing. Here's What That Actually Means.]]></title>
        <id>https://hostim.dev/blog/heroku-sustaining-model/</id>
        <link href="https://hostim.dev/blog/heroku-sustaining-model/"/>
        <updated>2026-02-12T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Heroku is moving to a sustaining engineering model. No new enterprise contracts, no big new features. Why this happens to VC-backed platforms – and why Hostim.dev is built differently.]]></summary>
        <content type="html"><![CDATA[<p>Heroku just said it's moving to a <strong>"sustaining engineering" model</strong>.</p>
<p>That's corporate speak for:</p>
<ul>
<li class="">No big new features</li>
<li class="">Focus on stability and security</li>
<li class="">No new enterprise contracts</li>
<li class="">Maintain what exists</li>
</ul>
<p>Heroku isn't shutting down.</p>
<p>But it's not a growth product anymore.</p>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-this-happens">Why This Happens<a href="https://hostim.dev/blog/heroku-sustaining-model/#why-this-happens" class="hash-link" aria-label="Direct link to Why This Happens" title="Direct link to Why This Happens" translate="no">​</a></h2>
<p>This isn't surprising.</p>
<p>Heroku was bought by Salesforce. It grew fast for years. Developers loved it. Enterprises signed contracts. But inside a big company, every product has to justify its budget.</p>
<p>If it doesn't fit the current strategy – AI, enterprise tooling, whatever leadership cares about now – it slides down the priority list.</p>
<p>That's how it usually goes:</p>
<ol>
<li class="">Growth</li>
<li class="">Monetization</li>
<li class="">Cost control</li>
<li class="">Maintenance</li>
</ol>
<p>Heroku just moved to step four.</p>
<p>It'll keep running.
It'll stay stable.
But it won't be where new ideas happen.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-vc-backed-platforms-change">Why VC-Backed Platforms Change<a href="https://hostim.dev/blog/heroku-sustaining-model/#why-vc-backed-platforms-change" class="hash-link" aria-label="Direct link to Why VC-Backed Platforms Change" title="Direct link to Why VC-Backed Platforms Change" translate="no">​</a></h2>
<p>This isn't only about Heroku.</p>
<p>A lot of developer platforms follow the same path:</p>
<ul>
<li class="">Raise money</li>
<li class="">Grow fast</li>
<li class="">Keep prices low to gain users</li>
<li class="">Capture market share</li>
<li class="">Then focus on margins</li>
</ul>
<p>And at some point, growth slows.</p>
<p>So things change:</p>
<ul>
<li class="">Pricing gets adjusted</li>
<li class="">Free tiers disappear</li>
<li class="">Roadmaps slow down</li>
<li class="">Enterprise rules tighten</li>
</ul>
<p>Not because the product failed.</p>
<p>But because the incentives changed.</p>
<p>And incentives drive everything.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="what-this-means-for-developers">What This Means for Developers<a href="https://hostim.dev/blog/heroku-sustaining-model/#what-this-means-for-developers" class="hash-link" aria-label="Direct link to What This Means for Developers" title="Direct link to What This Means for Developers" translate="no">​</a></h2>
<p>If you're already using Heroku, nothing breaks tomorrow.</p>
<p>But choosing a platform is a long-term decision. You're betting on where it's headed, not just where it is today.</p>
<p>And a platform in maintenance mode isn't building the next chapter.</p>
<p>So naturally people start asking:</p>
<ul>
<li class="">Will pricing stay predictable?</li>
<li class="">Will meaningful features ship?</li>
<li class="">Is this the start of a slow decline?</li>
</ul>
<p>That's why every time news like this drops, searches for "Heroku alternative" spike again.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-hostimdev-is-structured-differently">Why Hostim.dev Is Structured Differently<a href="https://hostim.dev/blog/heroku-sustaining-model/#why-hostimdev-is-structured-differently" class="hash-link" aria-label="Direct link to Why Hostim.dev Is Structured Differently" title="Direct link to Why Hostim.dev Is Structured Differently" translate="no">​</a></h2>
<p>Hostim.dev wasn't built to chase growth charts.</p>
<p>It's:</p>
<ul>
<li class=""><strong>Bootstrapped</strong></li>
<li class=""><strong>Small by design</strong></li>
<li class="">Focused only on <strong>Docker apps + built-in databases</strong></li>
</ul>
<p>No venture funding.
No board pushing for aggressive expansion.
No sudden shift toward whatever trend investors want next.</p>
<p>That changes the incentives.</p>
<p>The goal isn't hypergrowth.</p>
<p>It's staying stable and useful.</p>
<p>So the focus is simple:</p>
<ul>
<li class="">Predictable pricing</li>
<li class="">Clean Docker deploys</li>
<li class="">Built-in Postgres, MySQL, Redis, and volumes</li>
<li class="">No credits</li>
<li class="">No enterprise lock-in</li>
</ul>
<p>And no sudden freeze because strategy changed somewhere above the product team.</p>
<p>When you stay focused, you don't need dramatic pivots.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-bigger-pattern">The Bigger Pattern<a href="https://hostim.dev/blog/heroku-sustaining-model/#the-bigger-pattern" class="hash-link" aria-label="Direct link to The Bigger Pattern" title="Direct link to The Bigger Pattern" translate="no">​</a></h2>
<p>Cloud platforms go through cycles.</p>
<p>You've seen it before:</p>
<ul>
<li class="">Pricing overhauls</li>
<li class="">Free tiers removed</li>
<li class="">Feature roadmaps slowed</li>
<li class="">Products quietly put into maintenance</li>
</ul>
<p>It's not drama.</p>
<p>It's business math.</p>
<p>Big platforms answer to shareholders.
Small platforms answer to survival.</p>
<p>Hostim.dev is built to survive – not to flip or exit.</p>
<hr>
<p>If you want a simple way to run Docker apps without betting on a product in maintenance mode:</p>
<p>👉 <a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1" target="_blank" rel="noopener noreferrer"><b>Try Hostim.dev</b></a></p>
<p>Let's build tools that don't need to freeze to stay alive.</p>]]></content>
        <category label="heroku" term="heroku"/>
        <category label="paas" term="paas"/>
        <category label="startup" term="startup"/>
        <category label="cloud" term="cloud"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Bastion Host & GitHub Actions on Hostim.dev]]></title>
        <id>https://hostim.dev/blog/bastion-host-github-actions/</id>
        <link href="https://hostim.dev/blog/bastion-host-github-actions/"/>
        <updated>2026-01-08T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Recent Hostim.dev updates driven by customer requests: GitHub Actions deployments, SSH bastion host access, and flexible Docker CI/CD workflows.]]></summary>
        <content type="html"><![CDATA[<p>I haven't posted updates for a while, but several core features landed on Hostim.dev recently.</p>
<p>Instead of shipping from a fixed roadmap, I'm following <strong>support-driven (customer-driven) development</strong>: features move to the top of the queue once users actively need them.</p>
<p>Over the past month, this resulted in three practical additions around <strong>Docker CI/CD</strong>, <strong>GitHub Actions deployment</strong>, and <strong>secure bastion host access</strong>.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="github-actions-deploy-for-docker-apps">GitHub Actions deploy for Docker apps<a href="https://hostim.dev/blog/bastion-host-github-actions/#github-actions-deploy-for-docker-apps" class="hash-link" aria-label="Direct link to GitHub Actions deploy for Docker apps" title="Direct link to GitHub Actions deploy for Docker apps" translate="no">​</a></h2>
<p>Hostim.dev now supports <a class="" href="https://hostim.dev/docs/apps/github-actions/"><strong>GitHub Actions deployments</strong></a> out of the box.</p>
<p>You can trigger a deploy directly from GitHub Actions using a simple API call. This works well for common <strong>Docker CI/CD</strong> setups:</p>
<ul>
<li class="">Build and deploy on merge to <code>main</code></li>
<li class="">Restart an app after pushing a new Docker image</li>
<li class="">Manual deploys via <code>workflow_dispatch</code></li>
</ul>
<p>There's no OAuth and no hidden logic. Your workflow controls everything – branches, conditions, environments. Hostim only executes the requested action.</p>
<p>This is especially useful if you already run <strong>CI/CD with Docker</strong> and just want a clean deployment target.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="bastion-host-for-secure-shell-access-to-containers">Bastion host for secure shell access to containers<a href="https://hostim.dev/blog/bastion-host-github-actions/#bastion-host-for-secure-shell-access-to-containers" class="hash-link" aria-label="Direct link to Bastion host for secure shell access to containers" title="Direct link to Bastion host for secure shell access to containers" translate="no">​</a></h2>
<p>Each project now includes a built-in <a class="" href="https://hostim.dev/docs/services/bastion/"><strong>SSH bastion host</strong></a>.</p>
<p>If you're unfamiliar: <strong>a bastion host is a hardened entry point</strong> used to access private infrastructure without exposing services to the public internet.</p>
<p>On Hostim.dev, the bastion host allows you to open a shell into running apps:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">shell my-app</span><br></span></code></pre></div></div>
<p>This answers common questions like:</p>
<ul>
<li class=""><em>What is a bastion host used for?</em></li>
<li class=""><em>How do I securely SSH into containers?</em></li>
<li class=""><em>How can I debug a production Docker app without public access?</em></li>
</ul>
<p>Typical use cases:</p>
<ul>
<li class="">Debugging production issues</li>
<li class="">Running database migrations</li>
<li class="">Inspecting environment variables</li>
<li class="">Accessing internal services safely</li>
</ul>
<p>The bastion host is private, key-based, and isolated per project.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="custom-commands-for-docker-apps">Custom commands for Docker apps<a href="https://hostim.dev/blog/bastion-host-github-actions/#custom-commands-for-docker-apps" class="hash-link" aria-label="Direct link to Custom commands for Docker apps" title="Direct link to Custom commands for Docker apps" translate="no">​</a></h2>
<p>Apps can now override the container command.</p>
<p>This enables common Docker patterns such as:</p>
<ul>
<li class="">One image, multiple roles (web + worker)</li>
<li class="">Background jobs using the same Docker image</li>
<li class="">CI/CD pipelines that reuse images across environments</li>
</ul>
<p>This pairs naturally with <strong>Docker CI/CD pipelines</strong>, where images are built once and reused consistently.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-this-approach">Why this approach<a href="https://hostim.dev/blog/bastion-host-github-actions/#why-this-approach" class="hash-link" aria-label="Direct link to Why this approach" title="Direct link to Why this approach" translate="no">​</a></h2>
<p>Many platforms ship features based on assumptions.</p>
<p>Instead, these changes came directly from:</p>
<ul>
<li class="">"How do I deploy with GitHub Actions?"</li>
<li class="">"How do I get shell access without exposing ports?"</li>
<li class="">"How do I run workers with the same image?"</li>
</ul>
<p>Support questions shape the roadmap.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="whats-next">What's next<a href="https://hostim.dev/blog/bastion-host-github-actions/#whats-next" class="hash-link" aria-label="Direct link to What's next" title="Direct link to What's next" translate="no">​</a></h2>
<p>More items are planned, but user feedback decides the order.</p>
<p>If something feels missing, it's probably already on the list.</p>
<p>👉 <a href="https://hostim.dev/" target="_blank" rel="noopener noreferrer" class="">https://hostim.dev</a></p>]]></content>
        <category label="docker" term="docker"/>
        <category label="bastion-host" term="bastion-host"/>
        <category label="github-actions" term="github-actions"/>
        <category label="ci-cd" term="ci-cd"/>
        <category label="paas" term="paas"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Fixing host.docker.internal on Linux ]]></title>
        <id>https://hostim.dev/blog/fixing-host-docker-internal-linux/</id>
        <link href="https://hostim.dev/blog/fixing-host-docker-internal-linux/"/>
        <updated>2025-11-28T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Why host.docker.internal works on Mac/Windows but fails on Linux, and the one-line fix to solve it in Docker Compose.]]></summary>
        <content type="html"><![CDATA[<p>If you've ever moved a Docker project from a Mac to a Linux server, you've probably hit this error:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">Connection refused: host.docker.internal:3000</span><br></span></code></pre></div></div>
<p>On macOS and Windows, <code>host.docker.internal</code> is a magic DNS name that resolves to your host machine's IP address. It's incredibly useful for connecting containers to local databases or APIs running outside of Docker.</p>
<p>But on Linux? <strong>It doesn't exist by default.</strong></p>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why">Why?<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#why" class="hash-link" aria-label="Direct link to Why?" title="Direct link to Why?" translate="no">​</a></h2>
<p>On macOS and Windows, Docker runs inside a lightweight virtual machine. <code>host.docker.internal</code> is a helper to bridge the gap between that VM and your actual host OS.</p>
<p>On Linux, Docker runs natively. There is no VM. The "host" is just... the host. But because containers are isolated, they still don't know the host's IP address automatically.</p>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-fix">The Fix<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#the-fix" class="hash-link" aria-label="Direct link to The Fix" title="Direct link to The Fix" translate="no">​</a></h2>
<p>You don't need hacky scripts or hardcoded IPs. Docker 20.10+ supports a special <code>host-gateway</code> value.</p>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="in-docker-compose">In Docker Compose<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#in-docker-compose" class="hash-link" aria-label="Direct link to In Docker Compose" title="Direct link to In Docker Compose" translate="no">​</a></h3>
<p>Add <code>extra_hosts</code> to your service definition:</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">services</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">my-app</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">image</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> my</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">app</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">latest</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">extra_hosts</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"host.docker.internal:host-gateway"</span><br></span></code></pre></div></div>
<p>That's it. Now <code>host.docker.internal</code> will resolve to the host's Docker gateway IP (usually <code>172.17.0.1</code>), allowing your container to talk to services listening on the host.</p>
<blockquote>
<p>Important: make sure the service on the host is listening on <code>0.0.0.0</code> (or on the Docker bridge IP), not just <code>127.0.0.1</code>.
Otherwise the container can reach the host, but the host refuses the connection.</p>
</blockquote>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="in-docker-cli">In Docker CLI<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#in-docker-cli" class="hash-link" aria-label="Direct link to In Docker CLI" title="Direct link to In Docker CLI" translate="no">​</a></h3>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">docker run --add-host host.docker.internal:host-gateway my-image</span><br></span></code></pre></div></div>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="a-note-on-firewalls">A Note on Firewalls<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#a-note-on-firewalls" class="hash-link" aria-label="Direct link to A Note on Firewalls" title="Direct link to A Note on Firewalls" translate="no">​</a></h2>
<p>If it still doesn't work, check your firewall (UFW or iptables).
UFW may block forwarded traffic from Docker networks.</p>
<p>To allow traffic from the default Docker subnet:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">sudo ufw allow from 172.17.0.0/16</span><br></span></code></pre></div></div>
<p>This permits container → host connections without exposing the <code>docker0</code> interface itself.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="tired-of-networking-issues">Tired of Networking Issues?<a href="https://hostim.dev/blog/fixing-host-docker-internal-linux/#tired-of-networking-issues" class="hash-link" aria-label="Direct link to Tired of Networking Issues?" title="Direct link to Tired of Networking Issues?" translate="no">​</a></h2>
<p>Networking is the hardest part of self-hosting.</p>
<a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1&amp;compose=1" target="_blank" rel="noopener noreferrer" data-umami-event="blog_host_internal_cta"><span class="button button--primary">Deploy on Hostim.dev</span></a>
<p>At Hostim.dev, we handle the networking layer for you. Deploy your containers and let them talk to each other securely, without messing with <code>extra_hosts</code> or firewalls.</p>]]></content>
        <category label="docker" term="docker"/>
        <category label="linux" term="linux"/>
        <category label="networking" term="networking"/>
        <category label="troubleshooting" term="troubleshooting"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[A Better Umami Dashboard with Grafana]]></title>
        <id>https://hostim.dev/blog/umami-grafana-dashboard/</id>
        <link href="https://hostim.dev/blog/umami-grafana-dashboard/"/>
        <updated>2025-11-19T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[How we extended Umami analytics with a custom Grafana dashboard–heatmaps, moving averages, bot filtering, and more.]]></summary>
        <content type="html"><![CDATA[<p><img decoding="async" loading="lazy" alt="Umami + Grafana dashboard overview" src="https://hostim.dev/assets/images/umami-grafana-dashboard-389234883fca9ae1aa68ef26f41f13c5.png" width="1788" height="902" class="img_ev3q"></p>
<p>Umami is great. Lightweight, privacy-friendly, no cookies, no tracking drama.
We use it ourselves on Hostim.dev, and we ship a <strong>one-click Umami template</strong> for anyone who wants simple, privacy-focused analytics.</p>
<p>But once you start relying on analytics to make actual decisions, you hit the limits pretty quickly.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-the-default-umami-dashboard-wasnt-enough">Why the default Umami dashboard wasn't enough<a href="https://hostim.dev/blog/umami-grafana-dashboard/#why-the-default-umami-dashboard-wasnt-enough" class="hash-link" aria-label="Direct link to Why the default Umami dashboard wasn't enough" title="Direct link to Why the default Umami dashboard wasn't enough" translate="no">​</a></h2>
<p>Umami intentionally keeps things minimal, but some gaps become obvious:</p>
<ul>
<li class="">No clear view of <strong>when</strong> visitors peak during the day</li>
<li class="">Hard to isolate <strong>bots</strong> from real traffic</li>
<li class="">No <strong>moving averages</strong> or trend smoothing</li>
<li class="">No grouped referrers (e.g. "search", "LLM", "other")</li>
<li class="">Limited visibility into relationships between sessions and custom events</li>
</ul>
<p>None of this is criticism – Umami is intentionally simple.
But sometimes you want more resolution.</p>
<p>So I took the quickest path:</p>
<p><strong>Deploy Grafana → connect it to Umami's PostgreSQL → build a custom dashboard.</strong></p>
<p>This took maybe ten minutes and unlocked:</p>
<ul>
<li class=""><strong>Daily heatmap</strong> showing real traffic peaks</li>
<li class=""><strong>7-day moving averages</strong> for referrers</li>
<li class=""><strong>Qualified sessions</strong> (≥2 pageviews) to filter out most bots</li>
<li class=""><strong>Selectable custom events</strong></li>
<li class=""><strong>Raw stats</strong> for the selected period</li>
</ul>
<p>Suddenly Umami became "actionable" instead of just "nice".</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="how-to-connect-grafana-to-umamis-postgresql">How to connect Grafana to Umami's PostgreSQL<a href="https://hostim.dev/blog/umami-grafana-dashboard/#how-to-connect-grafana-to-umamis-postgresql" class="hash-link" aria-label="Direct link to How to connect Grafana to Umami's PostgreSQL" title="Direct link to How to connect Grafana to Umami's PostgreSQL" translate="no">​</a></h2>
<p>Inside Grafana:</p>
<p><strong>Configuration → Data sources → Add data source → PostgreSQL</strong></p>
<p>Fill in the credentials from your Umami database and save.</p>
<p>You can now import our dashboard:</p>
<p>👉 <a href="https://grafana.com/grafana/dashboards/24431" target="_blank" rel="noopener noreferrer" class=""><strong>Grafana.com Dashboard</strong></a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="try-it-yourself">Try it yourself<a href="https://hostim.dev/blog/umami-grafana-dashboard/#try-it-yourself" class="hash-link" aria-label="Direct link to Try it yourself" title="Direct link to Try it yourself" translate="no">​</a></h2>
<p>If you prefer to self-host on your own VPS, here is a complete Docker Compose stack for Umami, PostgreSQL, and Grafana.</p>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="full-docker-compose-stack">Full Docker Compose stack<a href="https://hostim.dev/blog/umami-grafana-dashboard/#full-docker-compose-stack" class="hash-link" aria-label="Direct link to Full Docker Compose stack" title="Direct link to Full Docker Compose stack" translate="no">​</a></h3>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">services</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">postgres</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">image</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> postgres</span><span class="token punctuation" style="color:#393A34">:</span><span class="token number" style="color:#36acaa">15</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">restart</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> always</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">environment</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">POSTGRES_USER</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> umami</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">POSTGRES_PASSWORD</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> umami_pass</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">POSTGRES_DB</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> umami</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">volumes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> postgres_data</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">/var/lib/postgresql/data</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">umami</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">image</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> ghcr.io/umami</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">software/umami</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">postgres</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">latest</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">restart</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> always</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">depends_on</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> postgres</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">environment</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">DATABASE_URL</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> postgres</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">//umami</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">umami_pass@postgres</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">5432/umami</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">DATABASE_TYPE</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> postgresql</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">APP_SECRET</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"replace_this_with_a_random_secret"</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">ports</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"127.0.0.1:3000:3000"</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">grafana</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">image</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> grafana/grafana</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">latest</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">restart</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> always</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">depends_on</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> postgres</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">environment</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> GF_SERVER_DOMAIN=grafana.example.com</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      GF_SERVER_ROOT_URL=https</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">//grafana.example.com</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">ports</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"127.0.0.1:3001:3000"</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">volumes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> grafana_data</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">/var/lib/grafana</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">volumes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">postgres_data</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  grafana_data</span><span class="token punctuation" style="color:#393A34">:</span><br></span></code></pre></div></div>
<p>Start everything:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">docker compose up -d</span><br></span></code></pre></div></div>
<p>Then:</p>
<ul>
<li class="">Umami → <a href="http://localhost:3000/" target="_blank" rel="noopener noreferrer" class="">http://localhost:3000</a></li>
<li class="">Grafana → <a href="http://localhost:3001/" target="_blank" rel="noopener noreferrer" class="">http://localhost:3001</a></li>
</ul>
<p>In Grafana, configure a PostgreSQL data source:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">Host: postgres</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Port: 5432</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">User: umami</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Password: umami_pass</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Database: umami</span><br></span></code></pre></div></div>
<p>Import the dashboard using the ID from Grafana.com.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="dont-want-to-run-a-server">Don't want to run a server?<a href="https://hostim.dev/blog/umami-grafana-dashboard/#dont-want-to-run-a-server" class="hash-link" aria-label="Direct link to Don't want to run a server?" title="Direct link to Don't want to run a server?" translate="no">​</a></h2>
<p>If you don't want to manage Docker, OS maintenance, or networking, you can deploy the <strong>same stack</strong> on Hostim.dev by simply pasting the Compose file above when creating a new project.</p>
<ul>
<li class="">Choose <strong>Paste Docker Compose</strong></li>
<li class="">Use the YAML from this section</li>
</ul>
<p>Hostim.dev will handle HTTPS, internal networking, logs, metrics, and persistence for all three services.</p>
<p>👉 <a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1&amp;compose=1" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console"><b>Try Hostim.dev – deploy the full stack without touching SSH</b></a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="already-running-umami-on-hostimdev">Already running Umami on Hostim.dev?<a href="https://hostim.dev/blog/umami-grafana-dashboard/#already-running-umami-on-hostimdev" class="hash-link" aria-label="Direct link to Already running Umami on Hostim.dev?" title="Direct link to Already running Umami on Hostim.dev?" translate="no">​</a></h2>
<p>If you deployed Umami using the one-click template, you don't need a new project. Just:</p>
<ol>
<li class="">Create a <strong>separate Grafana App</strong> in the <strong>same project</strong></li>
<li class="">Use the <code>grafana/grafana:latest</code> image</li>
<li class="">Add the existing <strong>Umami PostgreSQL</strong> as a Grafana data source</li>
<li class="">Import the dashboard JSON</li>
</ol>
<p>Both apps run on the same private project network, so they can communicate without exposing ports or adjusting firewall rules.</p>]]></content>
        <category label="umami" term="umami"/>
        <category label="grafana" term="grafana"/>
        <category label="analytics" term="analytics"/>
        <category label="docker" term="docker"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[MetalLB on Hetzner Dedicated with vSwitch]]></title>
        <id>https://hostim.dev/blog/metallb-hetzner-vswitch/</id>
        <link href="https://hostim.dev/blog/metallb-hetzner-vswitch/"/>
        <updated>2025-10-28T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Step-by-step guide to exposing Kubernetes LoadBalancer services using Hetzner vSwitch routed IPs.]]></summary>
        <content type="html"><![CDATA[<p>When running Kubernetes on Hetzner Dedicated, there is no cloud load balancer. But you <em>can</em> provide public LoadBalancer IPs by attaching a routed IP range to a vSwitch and letting MetalLB announce addresses over L2.</p>
<p>Our setup:</p>
<ul>
<li class="">Calico (VXLAN + WireGuard)</li>
<li class="">kube-proxy IPVS with strictARP</li>
<li class="">ingress-nginx for ingress traffic</li>
</ul>
<hr>
<p>The diagram below illustrates the traffic flow: MetalLB advertises a public VIP from one node at a time, ingress-nginx receives it, and traffic is forwarded to the application pod running anywhere in the cluster.</p>
<p><img decoding="async" loading="lazy" alt="MetalLB + Hetzner vSwitch topology" src="https://hostim.dev/assets/images/metallb-on-hetzner-b9840a0061bf34d509a9c50e436a5124.png" width="1008" height="498" class="img_ev3q"></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="1-assign-a-public-subnet-to-your-vswitch">1. Assign a public subnet to your vSwitch<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#1-assign-a-public-subnet-to-your-vswitch" class="hash-link" aria-label="Direct link to 1. Assign a public subnet to your vSwitch" title="Direct link to 1. Assign a public subnet to your vSwitch" translate="no">​</a></h2>
<p>Example routed block Hetzner provides:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">Subnet:     123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Gateway:    123.45.67.33</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Usable:     123.45.67.34–38</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Broadcast:  123.45.67.39</span><br></span></code></pre></div></div>
<p>Attach your dedicated servers to the vSwitch (VLAN ID e.g. 4000).</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="2-configure-vswitch-vlan-on-each-node">2. Configure vSwitch VLAN on each node<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#2-configure-vswitch-vlan-on-each-node" class="hash-link" aria-label="Direct link to 2. Configure vSwitch VLAN on each node" title="Direct link to 2. Configure vSwitch VLAN on each node" translate="no">​</a></h2>
<p>Each node gets a <strong>/32</strong> from the subnet – Hetzner routes the whole /29 to your server.</p>
<blockquote>
<p><strong>Important note on routing table IDs</strong></p>
<p>This guide uses routing table <strong>200</strong> as an example.</p>
<p>If you are running <strong>Cilium</strong>, avoid table <code>200</code>: Cilium currently flushes all routes in table 200 on startup, which breaks vSwitch routing.</p>
<p>For Cilium-based installations, <strong>any other unused routing table ID works</strong> (for example <code>201</code>, <code>300</code>, or <code>1001</code>).</p>
<p>Reference: <a href="https://github.com/cilium/cilium/issues/38531" target="_blank" rel="noopener noreferrer" class="">https://github.com/cilium/cilium/issues/38531</a></p>
</blockquote>
<p>Create <code>/etc/netplan/10-vlan-4000.yaml</code>:</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">network</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">version</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">2</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">renderer</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> networkd</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">vlans</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">vlan4000</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">id</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">4000</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">link</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> eno1</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">mtu</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">1400</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">addresses</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> 123.45.67.38/32 </span><span class="token comment" style="color:#999988;font-style:italic"># node-specific</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">routes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">to</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 0.0.0.0/0</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">via</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.33</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">on-link</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token boolean important" style="color:#36acaa">true</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">200</span><span class="token plain"> </span><span class="token comment" style="color:#999988;font-style:italic"># example table ID</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">to</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">scope</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> link</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">200</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token key atrule" style="color:#00a4db">routing-policy</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">from</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">200</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">priority</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">10</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">to</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">200</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">priority</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">10</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">from</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">to</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 10.233.0.0/18</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">254</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">priority</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">0</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">        </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token key atrule" style="color:#00a4db">from</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 123.45.67.32/29</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">to</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> 10.233.64.0/18</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">table</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">254</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">          </span><span class="token key atrule" style="color:#00a4db">priority</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token number" style="color:#36acaa">0</span><br></span></code></pre></div></div>
<p>Apply:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">netplan apply</span><br></span></code></pre></div></div>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="3-required-sysctl-settings">3. Required sysctl settings<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#3-required-sysctl-settings" class="hash-link" aria-label="Direct link to 3. Required sysctl settings" title="Direct link to 3. Required sysctl settings" translate="no">​</a></h2>
<p>Create <code>/etc/sysctl.d/999-metallb.conf</code>:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">net.ipv4.conf.all.arp_ignore=1</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">net.ipv4.conf.all.arp_announce=2</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">net.ipv4.conf.all.rp_filter=0</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">net.ipv4.conf.default.rp_filter=0</span><br></span></code></pre></div></div>
<p>Why:</p>
<table><thead><tr><th>Setting</th><th>Purpose</th></tr></thead><tbody><tr><td><code>arp_ignore=1</code></td><td>Only reply to ARP queries for an IP <strong>on the correct interface</strong> – prevents conflicting replies from Calico/VXLAN.</td></tr><tr><td><code>arp_announce=2</code></td><td>Send ARP only from the <strong>interface that owns the VIP</strong>, required when MetalLB moves VIPs between nodes.</td></tr><tr><td><code>rp_filter=0</code></td><td>Disable strict reverse-path filtering – otherwise nodes drop return traffic sourced from VIPs or remote pods.</td></tr></tbody></table>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="4-kube-proxy--calico-adjustments">4. kube-proxy + Calico adjustments<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#4-kube-proxy--calico-adjustments" class="hash-link" aria-label="Direct link to 4. kube-proxy + Calico adjustments" title="Direct link to 4. kube-proxy + Calico adjustments" translate="no">​</a></h2>
<p>Enable strictARP in kube-proxy (IPVS mode):</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">apiVersion</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> kubeproxy.config.k8s.io/v1alpha1</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">kind</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> KubeProxyConfiguration</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">ipvs</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">strictARP</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token boolean important" style="color:#36acaa">true</span><br></span></code></pre></div></div>
<p>MTU must account for VXLAN + vSwitch + WireGuard overhead:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">Calico MTU: 1280 (consistent across nodes)</span><br></span></code></pre></div></div>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="5-deploy-metallb">5. Deploy MetalLB<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#5-deploy-metallb" class="hash-link" aria-label="Direct link to 5. Deploy MetalLB" title="Direct link to 5. Deploy MetalLB" translate="no">​</a></h2>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">apiVersion</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> metallb.io/v1beta1</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">kind</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> IPAddressPool</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">metadata</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">name</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> vswitch</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">namespace</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> metallb</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">system</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">spec</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">addresses</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> 123.45.67.34</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">123.45.67.36 </span><span class="token comment" style="color:#999988;font-style:italic"># free VIPs</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token punctuation" style="color:#393A34">---</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">apiVersion</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> metallb.io/v1beta1</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">kind</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> L2Advertisement</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">metadata</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">name</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> l2</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">namespace</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> metallb</span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain">system</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">spec</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">ipAddressPools</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token punctuation" style="color:#393A34">[</span><span class="token string" style="color:#e3116c">"vswitch"</span><span class="token punctuation" style="color:#393A34">]</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">interfaces</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> </span><span class="token punctuation" style="color:#393A34">[</span><span class="token string" style="color:#e3116c">"vlan4000"</span><span class="token punctuation" style="color:#393A34">]</span><br></span></code></pre></div></div>
<p>Restart MetalLB speakers to pick up interface binding.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="6-ingress-service-configuration">6. Ingress service configuration<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#6-ingress-service-configuration" class="hash-link" aria-label="Direct link to 6. Ingress service configuration" title="Direct link to 6. Ingress service configuration" translate="no">​</a></h2>
<p>For ingress-nginx:</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">spec</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">externalTrafficPolicy</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> Local</span><br></span></code></pre></div></div>
<p>Pro:</p>
<ul>
<li class="">Preserves client IP</li>
<li class="">Prevents traffic hairpin across nodes</li>
</ul>
<p>Tradeoff:</p>
<ul>
<li class="">Only one node handles a given connection (acceptable for ingress)</li>
</ul>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="7-verification">7. Verification<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#7-verification" class="hash-link" aria-label="Direct link to 7. Verification" title="Direct link to 7. Verification" translate="no">​</a></h2>
<p>Confirm that your ingress-nginx Service received a public VIP:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">kubectl get svc -n ingress-nginx ingress-nginx-controller</span><br></span></code></pre></div></div>
<p>Expected example:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">NAME                       TYPE           CLUSTER-IP      EXTERNAL-IP    PORT(S)                      AGE</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">ingress-nginx-controller   LoadBalancer   10.233.53.156   123.45.67.35   80:30440/TCP,443:30477/TCP   17d</span><br></span></code></pre></div></div>
<p>Inspect the Service events to see which node currently advertises the VIP:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">kubectl describe svc -n ingress-nginx ingress-nginx-controller</span><br></span></code></pre></div></div>
<p>Look for:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">Events:</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  Normal  nodeAssigned  ...  metallb-speaker  announcing from node "control-plane-1" with protocol "layer2"</span><br></span></code></pre></div></div>
<p>Then verify reachability from <strong>outside</strong>:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">curl -I http://123.45.67.35</span><br></span></code></pre></div></div>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="failover-test">Failover test<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#failover-test" class="hash-link" aria-label="Direct link to Failover test" title="Direct link to Failover test" translate="no">​</a></h3>
<ol>
<li class="">Identify the active announcer from the above events</li>
<li class="">Shut that node down abruptly:</li>
</ol>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">sudo poweroff</span><br></span></code></pre></div></div>
<ol start="3">
<li class="">Re-run:</li>
</ol>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">curl -I http://123.45.67.35</span><br></span></code></pre></div></div>
<p>Expected: traffic continues within ~1–2 seconds as another node picks up the VIP.</p>
<p>➡️ Note: VIPs <strong>do not appear</strong> in <code>ip addr</code> on nodes; they are held in IPVS and advertised via ARP. That is normal.</p>
<hr>
<blockquote>
<p>Acknowledgment</p>
<p>Thanks to Oleksandr Vorona (DevOps at Dysnix) for reporting a Cilium routing table conflict and helping improve this guide.</p>
<p><a href="https://dysnix.com/" target="_blank" rel="noopener noreferrer" class="">https://dysnix.com</a></p>
</blockquote>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="wrapping-up">Wrapping up<a href="https://hostim.dev/blog/metallb-hetzner-vswitch/#wrapping-up" class="hash-link" aria-label="Direct link to Wrapping up" title="Direct link to Wrapping up" translate="no">​</a></h2>
<p>This gives:</p>
<ul>
<li class="">Public LoadBalancer IPs</li>
<li class="">Fast failover (~1-2s)</li>
<li class="">Clean separation: pod networking via VXLAN/WireGuard, external via vSwitch</li>
</ul>
<p>Alternatives:</p>
<ul>
<li class="">Hetzner Cloud Load Balancer (simpler, works with Dedicated too)</li>
<li class="">Cilium with L2 announcements</li>
</ul>
<p>We run hosting infrastructure, so controlling ingress networking ourselves matters (mostly to prove the point really). Hetzner still runs the vSwitch underneath, but it's more independent than relying on the cloud LB.</p>
<p>And if you'd rather not handle any of this yourself – <a href="https://console.hostim.dev/" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console"><b>Hostim.dev</b></a> is now live. You can deploy your Docker or Compose apps with built-in databases, volumes, HTTPS, and logs – all in one place, ready in minutes.</p>]]></content>
        <category label="metallb" term="metallb"/>
        <category label="kubernetes" term="kubernetes"/>
        <category label="hetzner" term="hetzner"/>
        <category label="vswitch" term="vswitch"/>
        <category label="networking" term="networking"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[How to Self-Host n8n with Docker Compose]]></title>
        <id>https://hostim.dev/blog/self-host-n8n-docker-compose/</id>
        <link href="https://hostim.dev/blog/self-host-n8n-docker-compose/"/>
        <updated>2025-10-04T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[A step-by-step guide to running n8n on a VPS with Docker Compose, making it survive reboots, and exposing it securely with Caddy.]]></summary>
        <content type="html"><![CDATA[<p><a href="https://n8n.io/" target="_blank" rel="noopener noreferrer" class="">n8n</a> is a popular open-source automation tool – like Zapier, but self-hosted.
Here's how to run it on your own VPS using Docker Compose, and expose it securely over HTTPS using <strong>Caddy</strong> as the ingress proxy.</p>
<p>If you want a broader primer, check out <a class="" href="https://hostim.dev/blog/how-to-self-host-docker-compose/">How to Self-Host a Docker Compose App</a>. But you don't need to read it first – this guide is self-contained.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="1-install-docker-on-your-vps">1. Install Docker on your VPS<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#1-install-docker-on-your-vps" class="hash-link" aria-label="Direct link to 1. Install Docker on your VPS" title="Direct link to 1. Install Docker on your VPS" translate="no">​</a></h2>
<p>Update the system and install Docker + Compose plugin:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt update &amp;&amp; apt upgrade -y</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get install ca-certificates curl -y</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">install -m 0755 -d /etc/apt/keyrings</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">chmod a+r /etc/apt/keyrings/docker.asc</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">echo \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  $(. /etc/os-release &amp;&amp; echo "${UBUNTU_CODENAME:-$VERSION_CODENAME}") stable" \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  | tee /etc/apt/sources.list.d/docker.list &gt; /dev/null</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get update</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get install docker-ce docker-ce-cli containerd.io docker-compose-plugin -y</span><br></span></code></pre></div></div>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="2-write-a-docker-compose-file">2. Write a Docker Compose file<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#2-write-a-docker-compose-file" class="hash-link" aria-label="Direct link to 2. Write a Docker Compose file" title="Direct link to 2. Write a Docker Compose file" translate="no">​</a></h2>
<p>Create a new directory for n8n and add <code>docker-compose.yml</code>:</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">services</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token key atrule" style="color:#00a4db">n8n</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">image</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> n8nio/n8n</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">latest</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">restart</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"> always</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">ports</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"127.0.0.1:5678:5678"</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">environment</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> N8N_HOST=n8n.example.com</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> N8N_PORT=5678</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> N8N_PROTOCOL=https</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    </span><span class="token key atrule" style="color:#00a4db">volumes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">      </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> n8n_data</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain">/home/node/.n8n</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"></span><span class="token key atrule" style="color:#00a4db">volumes</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  n8n_data</span><span class="token punctuation" style="color:#393A34">:</span><br></span></code></pre></div></div>
<blockquote>
<p>💡 Note: This guide assumes you already have a domain (like <code>n8n.example.com</code>) pointing to your VPS's IP address. If not, set that up with your DNS provider before continuing.</p>
</blockquote>
<p>Notice we bind to <code>127.0.0.1:5678</code> – so it's only accessible locally. Caddy will handle public access.</p>
<p>Start it:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">docker compose up -d</span><br></span></code></pre></div></div>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="3-make-it-survive-reboots">3. Make it survive reboots<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#3-make-it-survive-reboots" class="hash-link" aria-label="Direct link to 3. Make it survive reboots" title="Direct link to 3. Make it survive reboots" translate="no">​</a></h2>
<p>Create a systemd service:</p>
<div class="language-ini codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-ini codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain"># /etc/systemd/system/n8n.service</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Unit]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Description=n8n workflow automation (Docker Compose)</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">After=network.target</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Service]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Type=oneshot</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">WorkingDirectory=/root/n8n</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">ExecStart=/usr/bin/docker compose up -d</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">ExecStop=/usr/bin/docker compose down</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">RemainAfterExit=yes</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Install]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">WantedBy=multi-user.target</span><br></span></code></pre></div></div>
<p>Enable it:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">systemctl enable n8n</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">systemctl start n8n</span><br></span></code></pre></div></div>
<p>Now n8n restarts automatically after reboots.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="4-install-and-configure-caddy">4. Install and configure Caddy<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#4-install-and-configure-caddy" class="hash-link" aria-label="Direct link to 4. Install and configure Caddy" title="Direct link to 4. Install and configure Caddy" translate="no">​</a></h2>
<p>Caddy is a modern reverse proxy with <strong>automatic HTTPS</strong>. Perfect for small setups.</p>
<p>If you're curious about how Caddy compares to Nginx, HAProxy, or Traefik, see <a class="" href="https://hostim.dev/blog/reverse-proxy-showdown/">The Reverse Proxy Showdown</a>. For n8n, Caddy is the simplest choice.</p>
<p>Make sure your domain (e.g. <code>n8n.example.com</code>) already resolves to your VPS before you configure Caddy. Otherwise, Let's Encrypt won't be able to issue a certificate.</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt install -y debian-keyring debian-archive-keyring apt-transport-https</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | tee /etc/apt/trusted.gpg.d/caddy-stable.asc</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/debian.deb.txt' | tee /etc/apt/sources.list.d/caddy-stable.list</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt update</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt install caddy -y</span><br></span></code></pre></div></div>
<p>Edit the Caddyfile:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">nano /etc/caddy/Caddyfile</span><br></span></code></pre></div></div>
<p>Add:</p>
<div class="language-text codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-text codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">n8n.example.com {</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    reverse_proxy localhost:5678</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">}</span><br></span></code></pre></div></div>
<p>Reload Caddy:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">systemctl reload caddy</span><br></span></code></pre></div></div>
<p>That's it – Caddy requests and renews Let's Encrypt certificates automatically.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="5-secure-access">5. Secure access<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#5-secure-access" class="hash-link" aria-label="Direct link to 5. Secure access" title="Direct link to 5. Secure access" translate="no">​</a></h2>
<ul>
<li class="">Use strong credentials when creating first user.</li>
<li class="">Keep the <code>docker-compose.yml</code> volume so your workflows persist.</li>
<li class="">Optionally, restrict access to your IP range using Caddy if it's just for personal use.</li>
</ul>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="wrapping-up">Wrapping Up<a href="https://hostim.dev/blog/self-host-n8n-docker-compose/#wrapping-up" class="hash-link" aria-label="Direct link to Wrapping Up" title="Direct link to Wrapping Up" translate="no">​</a></h2>
<p>You now have a self-hosted <strong>n8n</strong> instance:</p>
<ul>
<li class="">Running in Docker Compose</li>
<li class="">Restarting automatically after reboots</li>
<li class="">Exposed via Caddy with HTTPS</li>
</ul>
<p>If you want to avoid managing servers altogether, platforms like <a class="" href="https://hostim.dev/blog/from-vps-to-paas/">Hostim.dev</a> let you paste a Compose file and get HTTPS, metrics, and persistence without touching SSH.</p>
<p>But if you prefer DIY – this setup will take you far.</p>]]></content>
        <category label="n8n" term="n8n"/>
        <category label="docker" term="docker"/>
        <category label="docker-compose" term="docker-compose"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
        <category label="tutorial" term="tutorial"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[The Reverse Proxy Showdown: Nginx vs HAProxy vs Caddy vs Traefik]]></title>
        <id>https://hostim.dev/blog/reverse-proxy-showdown/</id>
        <link href="https://hostim.dev/blog/reverse-proxy-showdown/"/>
        <updated>2025-09-16T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Nginx, HAProxy, Caddy, and Traefik are the top four reverse proxies. Each shines in different scenarios–here's how they compare, with pros, cons, and use cases.]]></summary>
        <content type="html"><![CDATA[<p>Reverse proxies are the unsung heroes of modern infrastructure. They terminate TLS, route traffic, balance loads, and keep your apps reachable. But which one should you choose?
There are four popular options worth comparing head-to-head: <strong>Nginx, HAProxy, Caddy, and Traefik</strong>. Each comes with its own strengths, trade-offs, and ideal use cases.</p>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="nginx--the-classic-all-rounder">Nginx – the classic all-rounder<a href="https://hostim.dev/blog/reverse-proxy-showdown/#nginx--the-classic-all-rounder" class="hash-link" aria-label="Direct link to Nginx – the classic all-rounder" title="Direct link to Nginx – the classic all-rounder" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> general-purpose web serving, static content, simple reverse proxy setups</li>
<li class=""><strong>Strengths:</strong> battle-tested, massive ecosystem, tons of tutorials, easy Certbot integration</li>
<li class=""><strong>Weaknesses:</strong> verbose configs, not as dynamic as newer tools</li>
</ul>
<p>Nginx is often the default choice. It's powerful, stable, and widely documented. If you're setting up a straightforward proxy or serving static files alongside your app, Nginx will feel familiar and reliable. Just be prepared to manage slightly more configuration boilerplate.</p>
<p>👉 <a class="" href="https://hostim.dev/learn/proxies/nginx/">Full Nginx reverse proxy guide →</a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="haproxy--the-performance-beast">HAProxy – the performance beast<a href="https://hostim.dev/blog/reverse-proxy-showdown/#haproxy--the-performance-beast" class="hash-link" aria-label="Direct link to HAProxy – the performance beast" title="Direct link to HAProxy – the performance beast" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> high-traffic sites, low-latency routing, advanced load balancing</li>
<li class=""><strong>Strengths:</strong> blazing fast, robust observability, flexible ACL system</li>
<li class=""><strong>Weaknesses:</strong> steeper learning curve, TLS setup can be fiddly</li>
</ul>
<p>HAProxy is famous for performance. It's a favorite in environments where uptime and throughput matter most. Think enterprise setups or any case where you need fine-grained control over routing logic and health checks. It's less beginner-friendly, but extremely powerful once mastered.</p>
<p>👉 <a class="" href="https://hostim.dev/learn/proxies/haproxy/">Full HAProxy reverse proxy guide →</a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="caddy--the-modern-batteries-included-choice">Caddy – the modern "batteries included" choice<a href="https://hostim.dev/blog/reverse-proxy-showdown/#caddy--the-modern-batteries-included-choice" class="hash-link" aria-label="Direct link to Caddy – the modern &quot;batteries included&quot; choice" title="Direct link to Caddy – the modern &quot;batteries included&quot; choice" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> minimal config, automatic HTTPS, developer-friendly defaults</li>
<li class=""><strong>Strengths:</strong> one-line proxy configs, TLS handled automatically, sane defaults</li>
<li class=""><strong>Weaknesses:</strong> smaller ecosystem, fewer advanced knobs for complex routing</li>
</ul>
<p>Caddy made waves by taking the pain out of HTTPS. With a simple <code>Caddyfile</code>, you get automatic TLS, redirects, and reverse proxying. It's ideal for small projects or developers who want secure, working defaults without fiddling with extra tooling.</p>
<p>👉 <a class="" href="https://hostim.dev/learn/proxies/caddy/">Full Caddy reverse proxy guide →</a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="traefik--the-container-native-router">Traefik – the container-native router<a href="https://hostim.dev/blog/reverse-proxy-showdown/#traefik--the-container-native-router" class="hash-link" aria-label="Direct link to Traefik – the container-native router" title="Direct link to Traefik – the container-native router" translate="no">​</a></h2>
<ul>
<li class=""><strong>Best for:</strong> Docker and Kubernetes workloads, dynamic environments</li>
<li class=""><strong>Strengths:</strong> integrates with container labels, dynamic service discovery, built-in metrics</li>
<li class=""><strong>Weaknesses:</strong> YAML configs can get verbose, less popular outside containerized setups</li>
</ul>
<p>Traefik was built with cloud-native apps in mind. Instead of editing config files, you annotate containers with labels and Traefik routes traffic automatically. It shines in environments where services come and go frequently, making it a natural fit for orchestrators like Kubernetes.</p>
<p>👉 <a class="" href="https://hostim.dev/learn/proxies/traefik/">Full Traefik reverse proxy guide →</a></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="quick-comparison">Quick comparison<a href="https://hostim.dev/blog/reverse-proxy-showdown/#quick-comparison" class="hash-link" aria-label="Direct link to Quick comparison" title="Direct link to Quick comparison" translate="no">​</a></h2>
<p><em>Looking for full setup walkthroughs? Check out our guides for <a class="" href="https://hostim.dev/learn/proxies/nginx/">Nginx</a>, <a class="" href="https://hostim.dev/learn/proxies/haproxy/">HAProxy</a>, <a class="" href="https://hostim.dev/learn/proxies/caddy/">Caddy</a>, and <a class="" href="https://hostim.dev/learn/proxies/traefik/">Traefik</a>.</em></p>
<table><thead><tr><th>Feature</th><th><strong>Nginx</strong></th><th><strong>HAProxy</strong></th><th><strong>Caddy</strong></th><th><strong>Traefik</strong></th></tr></thead><tbody><tr><td>Ease of setup</td><td>⭐⭐</td><td>⭐⭐</td><td>⭐⭐⭐⭐⭐</td><td>⭐⭐⭐⭐</td></tr><tr><td>Performance</td><td>⭐⭐⭐⭐</td><td>⭐⭐⭐⭐⭐</td><td>⭐⭐⭐</td><td>⭐⭐⭐⭐</td></tr><tr><td>Auto HTTPS</td><td>Needs Certbot</td><td>Manual + hooks</td><td>Built-in</td><td>Built-in</td></tr><tr><td>Container native</td><td>No</td><td>No</td><td>Somewhat</td><td>Yes</td></tr><tr><td>Ecosystem/docs</td><td>Huge</td><td>Mature ops-focused</td><td>Growing dev-focused</td><td>Strong in Docker/K8s space</td></tr></tbody></table>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="so-which-one-should-you-choose">So which one should you choose?<a href="https://hostim.dev/blog/reverse-proxy-showdown/#so-which-one-should-you-choose" class="hash-link" aria-label="Direct link to So which one should you choose?" title="Direct link to So which one should you choose?" translate="no">​</a></h2>
<ul>
<li class=""><strong>Just learning or running a blog?</strong> → <strong>Nginx</strong></li>
<li class=""><strong>Handling big traffic or need reliability?</strong> → <strong>HAProxy</strong></li>
<li class=""><strong>Want HTTPS with zero config?</strong> → <strong>Caddy</strong></li>
<li class=""><strong>Running Docker/Kubernetes?</strong> → <strong>Traefik</strong></li>
</ul>]]></content>
        <category label="reverse-proxy" term="reverse-proxy"/>
        <category label="nginx" term="nginx"/>
        <category label="haproxy" term="haproxy"/>
        <category label="caddy" term="caddy"/>
        <category label="traefik" term="traefik"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Netlify's New Credit Pricing: When Cloud Rent Comes Due]]></title>
        <id>https://hostim.dev/blog/netlify-credit-pricing/</id>
        <link href="https://hostim.dev/blog/netlify-credit-pricing/"/>
        <updated>2025-09-08T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Netlify just switched to credit-based pricing. Here's why VC-backed platforms raise prices, and why Hostim.dev–bootstrapped and lean–won't follow the same playbook.]]></summary>
        <content type="html"><![CDATA[<blockquote>
<p>📌 <em>Last week, I wrote about <a href="https://hostim.dev/blog/cloud-rent-in-action" target="_blank" rel="noopener noreferrer" class="">Cloud Rent in Action</a> – how layers of middlemen drive up the cost of running a simple SaaS stack. Netlify's new pricing update feels like the same story, playing out live.</em></p>
</blockquote>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="what-changed-at-netlify">What Changed at Netlify<a href="https://hostim.dev/blog/netlify-credit-pricing/#what-changed-at-netlify" class="hash-link" aria-label="Direct link to What Changed at Netlify" title="Direct link to What Changed at Netlify" translate="no">​</a></h2>
<p>Netlify <a href="https://www.netlify.com/blog/new-pricing-credits/" target="_blank" rel="noopener noreferrer" class="">just rolled out</a> a <strong>credit-based pricing model</strong>.</p>
<ul>
<li class="">New accounts are now required to buy credits.</li>
<li class="">Every deploy, function, or gigabyte of bandwidth consumes those credits.</li>
<li class="">When the credits run out, your projects pause until you top up.</li>
<li class="">Legacy users can stay on old plans for now, but the future is clear: credits are the new normal.</li>
</ul>
<p>On paper, this looks like a simplification. In reality, it's the next stage of <strong>cloud rent</strong>.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-netlify-had-to-change">Why Netlify Had to Change<a href="https://hostim.dev/blog/netlify-credit-pricing/#why-netlify-had-to-change" class="hash-link" aria-label="Direct link to Why Netlify Had to Change" title="Direct link to Why Netlify Had to Change" translate="no">​</a></h2>
<p>For years, companies like Netlify grew fast thanks to <strong>venture capital money</strong>. Investors subsidized growth: cheap plans, generous free tiers, and aggressive marketing. The mission was simple – capture the market at any cost.</p>
<p>That was the <strong>market expansion phase</strong>.
VCs were happy to foot the bill as long as user numbers climbed.</p>
<p>Now we're in the <strong>market exploration (or sustainability) phase</strong>. Investors want returns. And that means:</p>
<ul>
<li class="">Free tiers shrink</li>
<li class="">Simple flat plans get replaced with credit systems</li>
<li class="">Costs shift from VC wallets to developer wallets</li>
</ul>
<p>It's not that Netlify suddenly became greedy – it's that the VC playbook <em>always</em> ends this way. Rent has to be collected. And developers end up paying it.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-problem-with-credit-pricing">The Problem With Credit Pricing<a href="https://hostim.dev/blog/netlify-credit-pricing/#the-problem-with-credit-pricing" class="hash-link" aria-label="Direct link to The Problem With Credit Pricing" title="Direct link to The Problem With Credit Pricing" translate="no">​</a></h2>
<p>Credits sound neat – one bucket, one metric. But for most developers, they create more problems than they solve:</p>
<ul>
<li class=""><strong>Mental overhead</strong> – you're forced to budget not just money, but deploys and requests.</li>
<li class=""><strong>Unpredictable bills</strong> – a sudden spike in traffic can drain credits overnight.</li>
<li class=""><strong>Complexity creep</strong> – hosting a static site shouldn't require a calculator.</li>
</ul>
<p>This is exactly the dynamic I wrote about in my <strong>Cloud Rent</strong> post: when platforms optimize for investor returns instead of developer trust, pricing drifts away from simplicity and fairness.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-hostimdev-is-different">Why Hostim.dev Is Different<a href="https://hostim.dev/blog/netlify-credit-pricing/#why-hostimdev-is-different" class="hash-link" aria-label="Direct link to Why Hostim.dev Is Different" title="Direct link to Why Hostim.dev Is Different" translate="no">​</a></h2>
<p>Hostim.dev was built with a completely different philosophy.</p>
<ul>
<li class="">
<p><strong>Bootstrapped, not VC-funded</strong></p>
<p>No investors. No pressure to flip pricing later. No "growth at all costs" phase.</p>
</li>
<li class="">
<p><strong>Lean team</strong></p>
<p>Right now it's just me – the founder – building and running the platform. That means lower overhead and no bloated payroll to pass on to you.</p>
</li>
<li class="">
<p><strong>Fair pricing from the beginning</strong></p>
<p>Plans are simple, predictable, and surge-safe. No credits, no hidden meters, no surprise bills.</p>
</li>
<li class="">
<p><strong>Built for developers, not investors</strong></p>
<p>The focus is on usability and transparency. You don't need to rewire your workflow to fit a platform's billing quirks.</p>
</li>
</ul>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="cloud-rent-vs-developer-trust">Cloud Rent vs. Developer Trust<a href="https://hostim.dev/blog/netlify-credit-pricing/#cloud-rent-vs-developer-trust" class="hash-link" aria-label="Direct link to Cloud Rent vs. Developer Trust" title="Direct link to Cloud Rent vs. Developer Trust" translate="no">​</a></h2>
<p>So if last week's post was the theory, this week is the proof:
<strong>Cloud rent always comes due.</strong> Netlify's credits are just the latest example.</p>
<p>At Hostim.dev, we're building the opposite:</p>
<ul>
<li class="">Flat, predictable plans</li>
<li class="">No surprise charges</li>
<li class="">Databases, volumes, and apps as first-class citizens</li>
<li class="">A platform you can trust, built for developers, not for VCs</li>
</ul>
<hr>
<p>👉 <a href="https://hostim.dev/" target="_blank" rel="noopener noreferrer" class="">Try Hostim.dev today</a> – your first project is free for 5 days, with no credit card required.</p>]]></content>
        <category label="cloud" term="cloud"/>
        <category label="paas" term="paas"/>
        <category label="devops" term="devops"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="startup" term="startup"/>
        <category label="pricing" term="pricing"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Cloud Rent in Action: How €50 Turns Into €200+]]></title>
        <id>https://hostim.dev/blog/cloud-rent-in-action/</id>
        <link href="https://hostim.dev/blog/cloud-rent-in-action/"/>
        <updated>2025-09-03T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[A concrete breakdown of how much cloud rent you actually pay on AWS compared to bare metal and Hostim.dev.]]></summary>
        <content type="html"><![CDATA[<p>When you pay for cloud hosting, you're not just paying for compute.
You're paying <strong>rent</strong>. And it adds up fast.</p>
<p><img decoding="async" loading="lazy" alt="Cloud Rent comparison" src="https://hostim.dev/assets/images/cloud-rent-in-action-5229458084dbe249a1ddd6996b287591.png" width="845" height="355" class="img_ev3q"></p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="️-what-you-think-youre-paying-for">🏗️ What You Think You're Paying For<a href="https://hostim.dev/blog/cloud-rent-in-action/#%EF%B8%8F-what-you-think-youre-paying-for" class="hash-link" aria-label="Direct link to 🏗️ What You Think You're Paying For" title="Direct link to 🏗️ What You Think You're Paying For" translate="no">​</a></h2>
<p>Let's say you need a small SaaS backend:</p>
<ul>
<li class="">2 apps (API + worker)</li>
<li class="">1 Postgres database</li>
<li class="">1 Redis for caching</li>
<li class="">A few gigs of storage</li>
</ul>
<p>Pretty standard stack.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-what-it-costs-on-aws">💸 What It Costs on AWS<a href="https://hostim.dev/blog/cloud-rent-in-action/#-what-it-costs-on-aws" class="hash-link" aria-label="Direct link to 💸 What It Costs on AWS" title="Direct link to 💸 What It Costs on AWS" translate="no">​</a></h2>
<ul>
<li class=""><strong>EC2 (2× t3.medium)</strong> → €50 / mo</li>
<li class=""><strong>RDS Postgres (db.t3.small, 10GB)</strong> → €22 / mo</li>
<li class=""><strong>ElastiCache Redis (cache.t3.micro)</strong> → €10 / mo</li>
<li class=""><strong>EBS storage (100GB)</strong> → €10 / mo</li>
<li class=""><strong>Data transfer (200GB egress)</strong> → €9 / mo</li>
</ul>
<p><strong>Total: ~€101 / mo</strong></p>
<p>That's without backups, monitoring, or any extras.
And without any "friendly" PaaS markup on top.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-what-it-costs-on-bare-metal">🏠 What It Costs on Bare Metal<a href="https://hostim.dev/blog/cloud-rent-in-action/#-what-it-costs-on-bare-metal" class="hash-link" aria-label="Direct link to 🏠 What It Costs on Bare Metal" title="Direct link to 🏠 What It Costs on Bare Metal" translate="no">​</a></h2>
<p>Hetzner: 12 threads (read cores), 64GB RAM, 1TB SSD → <strong>€44 / mo</strong>.</p>
<p>You could run <em>dozens</em> of those same apps + databases on one machine.
But if you don't want to babysit it, you go through AWS – and suddenly you're paying <strong>2× more</strong> for the same outcome.</p>
<p>Also, there are risks of course, what if someone nukes datacenter? (Same applies to AWS though).</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-add-a-middleman">🧃 Add a Middleman<a href="https://hostim.dev/blog/cloud-rent-in-action/#-add-a-middleman" class="hash-link" aria-label="Direct link to 🧃 Add a Middleman" title="Direct link to 🧃 Add a Middleman" translate="no">​</a></h2>
<p>Now add a VC-backed PaaS that just resells AWS.
Nice UI, Heroku-like DX… but you're paying <strong>another ×2 markup</strong>.</p>
<p>Your ~€100 stack just became <strong>€200-250 / mo.</strong></p>
<p>That's <strong>cloud rent</strong>: the difference between the infra you're actually using and the layers of middlemen you're forced to pay.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-what-were-doing-instead">🌱 What We're Doing Instead<a href="https://hostim.dev/blog/cloud-rent-in-action/#-what-were-doing-instead" class="hash-link" aria-label="Direct link to 🌱 What We're Doing Instead" title="Direct link to 🌱 What We're Doing Instead" translate="no">​</a></h2>
<p>Hostim.dev cuts out the middle layers:</p>
<ul>
<li class="">Runs on <strong>bare metal in Germany</strong></li>
<li class="">Includes <strong>Postgres, MySQL, Redis, Volumes</strong> out of the box</li>
<li class=""><strong>Automatic HTTPS, metrics, logs</strong></li>
<li class=""><strong>Plan-based pricing</strong> (no surprise bills)</li>
<li class=""><strong>5-day free trial</strong> + always-free small tiers</li>
</ul>
<p>You still get the convenience of a PaaS.
But without subsidizing investors, shareholders, or cloud landlords.
Just me, your humble wannabe hoster.</p>
<h4 class="anchor anchorTargetStickyNavbar_Vzrq" id="so-how-much-would-that-exact-stack-cost-on-hostimdev">So how much would that exact stack cost on Hostim.dev?<a href="https://hostim.dev/blog/cloud-rent-in-action/#so-how-much-would-that-exact-stack-cost-on-hostimdev" class="hash-link" aria-label="Direct link to So how much would that exact stack cost on Hostim.dev?" title="Direct link to So how much would that exact stack cost on Hostim.dev?" translate="no">​</a></h4>
<p>That's <strong>€34 / mo</strong>.
Includes 2 App replicas, Postgres, Redis, and a 50GB volume – with HTTPS, metrics, and logs baked in.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-see-it-live">🚀 See It Live<a href="https://hostim.dev/blog/cloud-rent-in-action/#-see-it-live" class="hash-link" aria-label="Direct link to 🚀 See It Live" title="Direct link to 🚀 See It Live" translate="no">​</a></h2>
<p>I just shipped <strong>authless trials</strong>:
paste a <code>docker-compose.yml</code>, and you'll see your app running in seconds – no signup needed.</p>
<p>👉 <a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1&amp;compose=1" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console"><b>Try it now</b></a></p>]]></content>
        <category label="cloud" term="cloud"/>
        <category label="paas" term="paas"/>
        <category label="devops" term="devops"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="startup" term="startup"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[How We Built a PaaS with Go, Kubernetes, and React]]></title>
        <id>https://hostim.dev/blog/how-we-built-a-paas/</id>
        <link href="https://hostim.dev/blog/how-we-built-a-paas/"/>
        <updated>2025-08-26T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[A behind-the-scenes look at the stack powering Hostim.dev – from bare metal servers with Ansible and Kubespray, to a custom Go operator, schema-first backend, and a React + Ant Design frontend.]]></summary>
        <content type="html"><![CDATA[<p>Building a PaaS as a solo founder means making choices. Some deliberate, some accidental, all of them tradeoffs.</p>
<p>Every tool comes with pros and cons, and the deciding factor is usually the most expensive resource of all: <strong>time</strong>.</p>
<p>If I can get the job done with something I already know, I'll take that path. I'll learn new tools when the project pays for it. Until then, it's all about moving forward with what works.</p>
<p>Here's how Hostim.dev is put together today – the stack that runs every app, database, and service behind the scenes.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-infra-ansible--kubespray">🖥 Infra: Ansible + Kubespray<a href="https://hostim.dev/blog/how-we-built-a-paas/#-infra-ansible--kubespray" class="hash-link" aria-label="Direct link to 🖥 Infra: Ansible + Kubespray" title="Direct link to 🖥 Infra: Ansible + Kubespray" translate="no">​</a></h2>
<p>Before Kubernetes even comes into the picture, there's infrastructure to manage.</p>
<p>I've spent six years working with <strong>Ansible</strong>, so it was my first pick. Hostim.dev runs on <strong>bare metal servers</strong> – and the Kubernetes clusters on top of them are provisioned with <a href="https://github.com/kubernetes-sigs/kubespray" target="_blank" rel="noopener noreferrer" class="">Kubespray</a>, which itself is a set of Ansible playbooks.</p>
<p>That means everything integrates nicely:</p>
<ul>
<li class="">My own playbooks handle <strong>server lifecycle</strong> (deploy new users, rotate keys, manage credentials).</li>
<li class="">Kubespray handles <strong>cluster lifecycle</strong> (deploy, upgrade, or scale clusters).</li>
</ul>
<p>Could Terraform do this job too? Maybe. But I'd spend more time learning it than deploying clusters. That's the tradeoff.</p>
<p>The upside: <strong>flexibility</strong>. I can run only the parts I need, when I need them.
The downside: it's not centralized – I run playbooks from my own machine. If two people apply different changes at the same time, you could hit conflicts. For now, that's a human problem, and we'll solve it in a human way.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="️-the-kubernetes-operator">⚙️ The Kubernetes Operator<a href="https://hostim.dev/blog/how-we-built-a-paas/#%EF%B8%8F-the-kubernetes-operator" class="hash-link" aria-label="Direct link to ⚙️ The Kubernetes Operator" title="Direct link to ⚙️ The Kubernetes Operator" translate="no">​</a></h2>
<p>The heart of the platform is a <strong>custom operator</strong> written in Go with <a href="https://github.com/kubernetes-sigs/kubebuilder" target="_blank" rel="noopener noreferrer" class="">Kubebuilder</a>.</p>
<p>If you're not familiar: an operator is basically a program that runs in the cluster and ensures the "desired state" matches the "actual state."</p>
<p>Examples:</p>
<ul>
<li class="">Update an app's environment variables → operator notices, triggers a restart.</li>
<li class="">Create a new database → operator picks a server, provisions it, updates permissions, applies migrations.</li>
<li class="">Scale an app → operator reconciles replicas until the cluster matches your request.</li>
</ul>
<p>It also emits the <strong>events</strong> you see in the dashboard. The backend subscribes to them, stores them, and triggers UI updates so what you see is always fresh.</p>
<p>This piece does a lot – from managing app lifecycles to handling Redis and Postgres placements – and is probably the best candidate for open-sourcing later on. No fixed timeline yet, but it's on my mind.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-backend-api-schema-first">🔗 Backend API: Schema First<a href="https://hostim.dev/blog/how-we-built-a-paas/#-backend-api-schema-first" class="hash-link" aria-label="Direct link to 🔗 Backend API: Schema First" title="Direct link to 🔗 Backend API: Schema First" translate="no">​</a></h2>
<p>All the business logic sits in the backend. It's written in <strong>Go</strong>, with:</p>
<ul>
<li class=""><a href="https://github.com/gin-gonic/gin" target="_blank" rel="noopener noreferrer" class="">Gin</a> for the HTTP server</li>
<li class=""><a href="https://entgo.io/" target="_blank" rel="noopener noreferrer" class="">Ent</a> as the ORM</li>
<li class=""><a href="https://github.com/deepmap/oapi-codegen" target="_blank" rel="noopener noreferrer" class="">oapi-codegen</a> to generate code from an <strong>OpenAPI schema</strong></li>
</ul>
<p>The workflow looks like this:</p>
<ol>
<li class="">Write the OpenAPI schema first.</li>
<li class="">Generate the server interfaces with <code>oapi-codegen</code>.</li>
<li class="">Implement the interfaces manually.</li>
<li class="">Wire them up with Ent models and Kubernetes operator objects.</li>
</ol>
<p>It's not 100% smooth (Ent and oapi-codegen don't integrate perfectly, so there's some type conversion glue). But overall, it means less boilerplate and more consistency.</p>
<p>At the end of the day, three parts come together:</p>
<ul>
<li class=""><strong>K8s objects</strong> (via the operator's Go package)</li>
<li class=""><strong>Database code</strong> (via Ent)</li>
<li class=""><strong>HTTP API</strong> (via oapi-codegen)</li>
</ul>
<p>My job: glue them together and add business logic. Which is exactly what Hostim.dev runs on today.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-frontend-react--ant-design">🎨 Frontend: React + Ant Design<a href="https://hostim.dev/blog/how-we-built-a-paas/#-frontend-react--ant-design" class="hash-link" aria-label="Direct link to 🎨 Frontend: React + Ant Design" title="Direct link to 🎨 Frontend: React + Ant Design" translate="no">​</a></h2>
<p>This is where I had the least experience. A good friend helped me bootstrap the project, and I leaned on LLMs for some of the early decisions.</p>
<p>Framework of choice: <strong>React + TypeScript</strong>.
UI library: <strong>Ant Design (Antd)</strong> – suggested by an LLM, picked mostly on a gut call. It does the job.</p>
<p>Could I have picked Svelte or Vue or "framework XYZ" instead? Sure. But I had a friend to guide me through React, not those other frameworks. And that meant I could start shipping right away. That's the tradeoff.</p>
<p>What I'm happy about is how code generation carries through to the frontend. The OpenAPI client is autogenerated, so the frontend just calls strongly typed functions.</p>
<p>If I change an object in the backend and re-generate, any breaking changes are immediately visible in the IDE. That feedback loop saved me a ton of time and bugs.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="wrapping-up">Wrapping Up<a href="https://hostim.dev/blog/how-we-built-a-paas/#wrapping-up" class="hash-link" aria-label="Direct link to Wrapping Up" title="Direct link to Wrapping Up" translate="no">​</a></h2>
<p>So that's the stack:</p>
<ul>
<li class=""><strong>Ansible + Kubespray</strong> for infra</li>
<li class=""><strong>Go + Kubebuilder</strong> operator for apps, DBs, Redis, volumes</li>
<li class=""><strong>Go + Gin + Ent + OpenAPI</strong> for backend</li>
<li class=""><strong>React + Ant Design</strong> for frontend</li>
</ul>
<p>It's not perfect. Every layer has its tradeoffs. Some tools might be "hotter" or "easier," but experience and context matter more. In the end, the real problem to solve isn't "which framework is coolest" – it's time.</p>
<p>That's true for me building the platform, and it's true for anyone using Hostim.dev instead of wiring up VPS configs or AWS bills.</p>
<p>Think of it like this: you can choose Terraform vs. Ansible, React vs. Vue… and you can also choose "Do I spend Saturday night fixing SSL, or do I just deploy and move on?" 😅</p>
<p>👉 If you want to try it out, click here: <a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1" target="_blank" rel="noopener noreferrer"><b>hostim.dev</b></a></p>]]></content>
        <category label="paas" term="paas"/>
        <category label="kubernetes" term="kubernetes"/>
        <category label="go" term="go"/>
        <category label="react" term="react"/>
        <category label="devops" term="devops"/>
        <category label="selfhosting" term="selfhosting"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[From VPS to PaaS: Why I Stopped Managing Servers]]></title>
        <id>https://hostim.dev/blog/from-vps-to-paas/</id>
        <link href="https://hostim.dev/blog/from-vps-to-paas/"/>
        <updated>2025-08-19T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Most side projects start with a VPS and Docker Compose. Cheap and simple – until it isn't. Here's why I switched to PaaS, and what I built to make it easier.]]></summary>
        <content type="html"><![CDATA[<p>Most side projects start the same way.
You grab a VPS from Hetzner or DigitalOcean, install Docker, run <code>docker compose up</code>, and boom – you're live.</p>
<p>It feels cheap. It feels simple.
Until it isn't.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-vps-path-the-default-way">The VPS Path: The Default Way<a href="https://hostim.dev/blog/from-vps-to-paas/#the-vps-path-the-default-way" class="hash-link" aria-label="Direct link to The VPS Path: The Default Way" title="Direct link to The VPS Path: The Default Way" translate="no">​</a></h2>
<p>Here's the typical journey I went through (and many devs still do):</p>
<ol>
<li class="">Rent a VPS for €5–€10/month</li>
<li class="">Install Docker + Docker Compose</li>
<li class="">Run the app</li>
<li class="">Add Nginx and Let's Encrypt for HTTPS</li>
<li class="">Hack together a systemd unit so it restarts after reboot</li>
<li class="">Manually configure backups, logs, and monitoring</li>
</ol>
<p>It works. But every new project means repeating the same steps.
And every time, something goes wrong – ports left open, SSL renewal fails, or a config breaks after an update.</p>
<p>Well, unless you properly automate it with something like <strong>Ansible</strong> or <strong>Terraform</strong>.
But let's be honest: do you really want to learn and maintain infra-as-code pipelines… just for side projects?</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-hidden-costs-of-cheap-vps-hosting">The Hidden Costs of "Cheap" VPS Hosting<a href="https://hostim.dev/blog/from-vps-to-paas/#the-hidden-costs-of-cheap-vps-hosting" class="hash-link" aria-label="Direct link to The Hidden Costs of &quot;Cheap&quot; VPS Hosting" title="Direct link to The Hidden Costs of &quot;Cheap&quot; VPS Hosting" translate="no">​</a></h2>
<p>At first glance, VPS looks cheap.
But the costs sneak up on you:</p>
<ul>
<li class=""><strong>Backups</strong>: €2–€5/month</li>
<li class=""><strong>Monitoring/logs</strong>: another €5–€10/month or DIY time</li>
<li class=""><strong>Downtime</strong>: hours spent debugging instead of coding</li>
<li class=""><strong>Security</strong>: one misconfigured firewall can expose your database</li>
</ul>
<p>The real cost isn't just money.
It's <strong>time lost</strong> repeating setup, patching servers, and fixing mistakes.
And if you're billing clients? That "cheap" VPS suddenly isn't cheap anymore.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="the-paas-alternative">The PaaS Alternative<a href="https://hostim.dev/blog/from-vps-to-paas/#the-paas-alternative" class="hash-link" aria-label="Direct link to The PaaS Alternative" title="Direct link to The PaaS Alternative" translate="no">​</a></h2>
<p>A PaaS (Platform-as-a-Service) takes that whole messy checklist and bakes it in:</p>
<ul>
<li class="">Deploy directly from <strong>Docker, Git, or Compose</strong></li>
<li class=""><strong>Automatic HTTPS</strong> and domain management</li>
<li class=""><strong>Built-in databases</strong> like Postgres, MySQL, and Redis</li>
<li class=""><strong>Volumes</strong> that survive restarts and redeploys</li>
<li class=""><strong>Metrics and logs</strong> out of the box</li>
<li class=""><strong>Per-project isolation</strong> so one client doesn't mess up another</li>
</ul>
<p>Instead of spending hours setting up a VPS, you paste your Compose file or point to a repo, click deploy, and it just works.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="why-i-switched-and-why-i-built-hostimdev">Why I Switched (and Why I Built Hostim.dev)<a href="https://hostim.dev/blog/from-vps-to-paas/#why-i-switched-and-why-i-built-hostimdev" class="hash-link" aria-label="Direct link to Why I Switched (and Why I Built Hostim.dev)" title="Direct link to Why I Switched (and Why I Built Hostim.dev)" translate="no">​</a></h2>
<p>After doing the VPS setup dozens of times – for my own apps, side projects, and client work – I finally hit a wall.</p>
<p>Every project felt like déjà vu.
Spin up server, fight configs, add SSL, fix logging, repeat.</p>
<p>So I built something that skips all of that.</p>
<p>Hostim.dev is a <strong>developer-first PaaS</strong>.
You paste your <code>docker-compose.yml</code>, or deploy from Git or Docker Hub, and you're live with HTTPS, metrics, databases, and volumes.</p>
<p>No YAML rewrites. No hidden cloud costs.
Just deploy and move on.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="wrapping-up">Wrapping Up<a href="https://hostim.dev/blog/from-vps-to-paas/#wrapping-up" class="hash-link" aria-label="Direct link to Wrapping Up" title="Direct link to Wrapping Up" translate="no">​</a></h2>
<p>VPS hosting isn't bad. It's still a good choice if you want full control or you enjoy tweaking configs.</p>
<p>But if you'd rather spend time building apps instead of babysitting servers, a PaaS can save you both money and frustration.</p>
<p>And yes – I'll still be babysitting servers.
But that's my job now, not yours. 😉</p>
<p>👉 <a href="https://console.hostim.dev/" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console"><b>Hostim.dev</b></a> is opening soon with a free trial and always-free database tiers.
If you're tired of fighting servers, <a class="" href="https://hostim.dev/">join the waitlist</a> – and let's bring hosting back to earth.</p>]]></content>
        <category label="vps" term="vps"/>
        <category label="paas" term="paas"/>
        <category label="docker" term="docker"/>
        <category label="docker-compose" term="docker-compose"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Lessons from 50 Devs on Docker Hosting]]></title>
        <id>https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/</id>
        <link href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/"/>
        <updated>2025-08-11T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Key takeaways from dozens of developer interviews on how they deploy projects, what they care about in hosting, and why some features became no-brainers for Hostim.dev.]]></summary>
        <content type="html"><![CDATA[<p>When you talk to enough developers about how they deploy projects, a few patterns start to emerge. Some are obvious in hindsight, others caught me completely off guard.</p>
<p>Here are my biggest takeaways so far.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="1-keep-talking-to-users--always">1. Keep Talking to Users – Always<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#1-keep-talking-to-users--always" class="hash-link" aria-label="Direct link to 1. Keep Talking to Users – Always" title="Direct link to 1. Keep Talking to Users – Always" translate="no">​</a></h3>
<p>Interviews aren't just a pre-launch thing. They work for <em>any</em> kind of app.</p>
<p>It's amazing how something that feels crystal clear to you as the builder can be completely unintuitive to someone else. Watching real people click around your UI will surface more "aha" moments than weeks of theorizing.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="2-interfaces-should-feel-alive">2. Interfaces Should Feel Alive<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#2-interfaces-should-feel-alive" class="hash-link" aria-label="Direct link to 2. Interfaces Should Feel Alive" title="Direct link to 2. Interfaces Should Feel Alive" translate="no">​</a></h3>
<p>Users want to feel in control and know what's happening.</p>
<p>If something is in progress, show it – a spinner, a loading bar, anything. If you can't give immediate results, fill the gap with meaningful feedback. Never leave people wondering, <em>"Is this thing stuck?"</em></p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="3-pretty-vs-functional-where-devs-lean">3. Pretty vs. Functional: Where Devs Lean<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#3-pretty-vs-functional-where-devs-lean" class="hash-link" aria-label="Direct link to 3. Pretty vs. Functional: Where Devs Lean" title="Direct link to 3. Pretty vs. Functional: Where Devs Lean" translate="no">​</a></h3>
<p>Maybe it's selection bias, but most developers I talked to care far more about a UI being <em>clear and functional</em> than it being flashy.</p>
<p>When AWS's interface is slow and clunky, anything even marginally better feels like a big improvement.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="4-your-landing-page-might-matter-less-than-you-think">4. Your Landing Page Might Matter Less Than You Think<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#4-your-landing-page-might-matter-less-than-you-think" class="hash-link" aria-label="Direct link to 4. Your Landing Page Might Matter Less Than You Think" title="Direct link to 4. Your Landing Page Might Matter Less Than You Think" translate="no">​</a></h3>
<p>Only a small fraction of devs I spoke to read landing pages in full. Many go straight to <strong>Getting Started</strong> or <strong>Try Now</strong>.</p>
<p>A common journey seems to be:
<strong>Top of page → Pricing → Try Now.</strong></p>
<p>Selection bias? Possibly. But it makes me think the "shiny" part of the landing page matters less than making the <em>first click</em> effortless.</p>
<p>If you do read this whole post, I'd love your thoughts on our landing page – what works, what doesn't, and what's missing. You can email me at <a href="mailto:pv@hostim.dev" target="_blank" rel="noopener noreferrer" class="">pv@hostim.dev</a>. Honest, constructive feedback is always welcome.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="5-many-devs-dont-even-know-the-big-paas-players">5. Many Devs Don't Even Know the "Big" PaaS Players<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#5-many-devs-dont-even-know-the-big-paas-players" class="hash-link" aria-label="Direct link to 5. Many Devs Don't Even Know the &quot;Big&quot; PaaS Players" title="Direct link to 5. Many Devs Don't Even Know the &quot;Big&quot; PaaS Players" translate="no">​</a></h3>
<p>This one surprised me at first: about half of the devs I spoke to didn't know the names of popular PaaS competitors.</p>
<p>In hindsight, it makes sense:</p>
<ul>
<li class="">At work, devs often don't handle deployments at all.</li>
<li class="">If they do, it's usually Kubernetes – and where it's hosted is someone else's problem.</li>
<li class="">For hobby projects, most people just rent a VPS, install Docker, and run <code>docker compose up</code>.</li>
</ul>
<p>The upside? There's still a lot of awareness to be built. The market isn't as saturated as it sometimes feels from the inside.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="6-listening-pays-off--literally-in-features">6. Listening Pays Off – Literally in Features<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#6-listening-pays-off--literally-in-features" class="hash-link" aria-label="Direct link to 6. Listening Pays Off – Literally in Features" title="Direct link to 6. Listening Pays Off – Literally in Features" translate="no">​</a></h3>
<p>When I first started sketching out my platform idea, it was going to be "bare" PaaS – you'd have to create apps, databases, and volumes yourself, wire them up, copy env vars around, etc.</p>
<p>Two questions kept coming up over and over:</p>
<ul>
<li class="">"Do you support Docker Compose?"</li>
<li class="">"Do you have templates?"</li>
</ul>
<p>At first, both answers were <em>no</em>. But after hearing it enough times, I built them.</p>
<ul>
<li class=""><strong>Templates</strong> now include <a class="" href="https://hostim.dev/docs/getting-started/app-stack/">five common stacks</a> – Spring Boot, Rails, FastAPI, Django, and Node – plus a bunch of <a class="" href="https://hostim.dev/docs/templates/">open-source apps</a> like Umami, Ghost, Actual Budget, Memos, and more.</li>
<li class=""><strong><a class="" href="https://hostim.dev/docs/getting-started/templates/#importing-with-docker-compose">Docker Compose</a> support</strong> takes your YAML and turns it into a template. If your Compose file builds from source, the platform will just ask for the Git repo and build it for you.</li>
</ul>
<p>Strong signals from user conversations made those features obvious to prioritize.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="wrapping-up">Wrapping Up<a href="https://hostim.dev/blog/what-i-learned-from-talking-to-50-devs/#wrapping-up" class="hash-link" aria-label="Direct link to Wrapping Up" title="Direct link to Wrapping Up" translate="no">​</a></h3>
<p>If you take one thing from this: talk to users early and often. Even if you think you <em>know</em> what they need – you don't, not until you watch them try it.</p>
<p>👉 <a href="https://console.hostim.dev/" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console">Get started with Hostim.dev</a></p>]]></content>
        <category label="docker" term="docker"/>
        <category label="paas" term="paas"/>
        <category label="devops" term="devops"/>
        <category label="interviews" term="interviews"/>
        <category label="product" term="product"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[How to Self-Host a Docker Compose App]]></title>
        <id>https://hostim.dev/blog/how-to-self-host-docker-compose/</id>
        <link href="https://hostim.dev/blog/how-to-self-host-docker-compose/"/>
        <updated>2025-08-04T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[A step-by-step guide to deploying your Docker Compose project on a VPS – and why I got tired of doing it manually.]]></summary>
        <content type="html"><![CDATA[<p>You've got a working <code>docker-compose.yml</code>, and now you want to put it online.</p>
<p>Maybe it's a SaaS side project. A personal site. A dashboard for a client. Whatever it is – you're here because you want to host a Compose app, and you don't want to spend hours fiddling with YAML, CI pipelines, or Kubernetes manifests.</p>
<p>Let's walk through what it really takes to host a Docker Compose project on your own. And then I'll show you what I built to make this process go away – for myself and anyone else who's tired of copy-pasting configs.</p>
<hr>
<blockquote>
<p><strong>Example:</strong> We'll use this project as our demo:
<a href="https://github.com/hostimdev/demo-django" target="_blank" rel="noopener noreferrer" class="">hostimdev/demo-django</a>
(A simple Django app with MySQL and Redis)</p>
</blockquote>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-the-hard-way-vps--docker--compose">🧱 The Hard Way: VPS + Docker + Compose<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#-the-hard-way-vps--docker--compose" class="hash-link" aria-label="Direct link to 🧱 The Hard Way: VPS + Docker + Compose" title="Direct link to 🧱 The Hard Way: VPS + Docker + Compose" translate="no">​</a></h2>
<p>First, the classic method. Take your favorite VPS provider (we like Hetzner – in fact, Hostim.dev runs on their bare metal servers), and spin up a server.</p>
<blockquote>
<p>Note: This guide assumes you're logged in as root.
If not, prefix commands with <code>sudo</code> or use <code>sudo -i</code> to switch to root.</p>
</blockquote>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="1-provision-the-vps">1. Provision the VPS<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#1-provision-the-vps" class="hash-link" aria-label="Direct link to 1. Provision the VPS" title="Direct link to 1. Provision the VPS" translate="no">​</a></h3>
<p>Pick a Linux image (Ubuntu or Debian), log in via SSH, and update your system:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt update &amp;&amp; apt upgrade -y</span><br></span></code></pre></div></div>
<p>Install Docker and docker-compose (we follow the <a href="https://docs.docker.com/engine/install/ubuntu/#install-using-the-repository" target="_blank" rel="noopener noreferrer" class="">official guide</a>):</p>
<ol>
<li class="">Set up Docker's apt repository.</li>
</ol>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain"># Add Docker's official GPG key:</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get update</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get install ca-certificates curl</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">install -m 0755 -d /etc/apt/keyrings</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">chmod a+r /etc/apt/keyrings/docker.asc</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain"># Add the repository to Apt sources:</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">echo \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  $(. /etc/os-release &amp;&amp; echo "${UBUNTU_CODENAME:-$VERSION_CODENAME}") stable" | \</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  tee /etc/apt/sources.list.d/docker.list &gt; /dev/null</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">apt-get update</span><br></span></code></pre></div></div>
<ol start="2">
<li class="">Install Docker packages:</li>
</ol>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin</span><br></span></code></pre></div></div>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="2-clone-your-project">2. Clone your project<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#2-clone-your-project" class="hash-link" aria-label="Direct link to 2. Clone your project" title="Direct link to 2. Clone your project" translate="no">​</a></h3>
<p>We'll use a <a href="https://github.com/hostimdev/demo-django" target="_blank" rel="noopener noreferrer" class="">demo Django app</a></p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">git clone https://github.com/hostimdev/demo-django.git</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">cd demo-django</span><br></span></code></pre></div></div>
<blockquote>
<p>💡 <strong>Private repo?</strong>
You can:</p>
<ul>
<li class="">Use a <a href="https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/creating-a-personal-access-token" target="_blank" rel="noopener noreferrer" class="">Personal Access Token (PAT)</a> and clone via HTTPS, or</li>
<li class="">Upload your VPS's <strong>public SSH key</strong> to GitHub and clone via SSH.</li>
</ul>
</blockquote>
<blockquote>
<p>⚠️ If you go with the SSH method, make sure to secure the private key on the VPS and limit server access. It's secure <em>if</em> your system is.</p>
</blockquote>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="3-deploy-your-app">3. Deploy your app<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#3-deploy-your-app" class="hash-link" aria-label="Direct link to 3. Deploy your app" title="Direct link to 3. Deploy your app" translate="no">​</a></h3>
<p>Assuming you have a <code>docker-compose.yml</code> ready:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">docker compose up -d</span><br></span></code></pre></div></div>
<p>That runs the app. But there are some problems:</p>
<ul>
<li class="">It won't restart after reboot.</li>
<li class="">There's no HTTPS.</li>
<li class="">You're exposing raw ports to the world.</li>
</ul>
<blockquote>
<p><strong>Security Note:</strong> To prevent exposing services to the public internet, modify your <code>docker-compose.yml</code> to bind ports only to localhost. For example:</p>
<div class="language-yaml codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-yaml codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token key atrule" style="color:#00a4db">ports</span><span class="token punctuation" style="color:#393A34">:</span><span class="token plain"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">  </span><span class="token punctuation" style="color:#393A34">-</span><span class="token plain"> </span><span class="token string" style="color:#e3116c">"127.0.0.1:8000:8000"</span><br></span></code></pre></div></div>
<p>This ensures services are only accessible from the local machine, not directly from the internet.</p>
</blockquote>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="4-add-https-with-nginx">4. Add HTTPS with nginx<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#4-add-https-with-nginx" class="hash-link" aria-label="Direct link to 4. Add HTTPS with nginx" title="Direct link to 4. Add HTTPS with nginx" translate="no">​</a></h3>
<p>Install nginx:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt install nginx -y</span><br></span></code></pre></div></div>
<p>Change the default config to proxy traffic to your app (assumes it runs on port 8000):</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">nano /etc/nginx/sites-available/default</span><br></span></code></pre></div></div>
<p>Replace the <code>location /</code> block inside the <code>server {}</code> with:</p>
<div class="language-nginx codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-nginx codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">location / {</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    proxy_pass http://localhost:8000;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    proxy_set_header Host $host;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    proxy_set_header X-Real-IP $remote_addr;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    proxy_set_header X-Forwarded-Proto $scheme;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">}</span><br></span></code></pre></div></div>
<p>Test and restart nginx:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">nginx -t</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">systemctl restart nginx</span><br></span></code></pre></div></div>
<p>Install Certbot and request an HTTPS certificate with automatic redirect:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">apt install certbot python3-certbot-nginx -y</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">certbot --nginx # follow the instructions</span><br></span></code></pre></div></div>
<blockquote>
<p>It will configure the https for you</p>
</blockquote>
<details class="details_lb9f alert alert--info details_b_Ee" data-collapsed="true"><summary>🔒 Bonus: Block direct access to your server's IP</summary><div><div class="collapsibleContent_i85q"><p>By default, if someone enters your server's IP address in a browser, nginx may respond with your app or a default welcome page. To prevent this and <strong>serve content only under your domain</strong>, block IP-based access.</p><p>Install <code>ssl-cert</code> package, we need it just for dummy certs:</p><div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">app install ssl-cert -y</span><br></span></code></pre></div></div><p>Edit your nginx config:</p><div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">nano /etc/nginx/sites-available/default</span><br></span></code></pre></div></div><p>Replace the <strong>topmost server block</strong> (usually the default one on port 80) with this:</p><div class="language-nginx codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-nginx codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain"># Block HTTP requests to IP (default server)</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">server {</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    listen 80 default_server;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    listen [::]:80 default_server;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    server_name _;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    return 444;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">}</span><br></span></code></pre></div></div><p>Then add this <strong>just below</strong> to block HTTPS access by IP:</p><div class="language-nginx codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-nginx codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain"># Block HTTPS requests to IP (default server)</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">server {</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    listen 443 ssl default_server;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    listen [::]:443 ssl default_server;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    server_name _;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    ssl_certificate /etc/ssl/certs/ssl-cert-snakeoil.pem;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    ssl_certificate_key /etc/ssl/private/ssl-cert-snakeoil.key;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">    return 444;</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">}</span><br></span></code></pre></div></div><blockquote>
<p>⚠️ Replace the cert paths with your real SSL cert/key if you're not using the default snakeoil test cert.</p>
</blockquote><p>Restart nginx:</p><div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">nginx -t</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">systemctl restart nginx</span><br></span></code></pre></div></div><p>From now on, only your <strong>domain name</strong> will serve content. Requests to the raw IP will be silently dropped (code 444 = connection closed with no response).</p></div></div></details>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="5-make-it-survive-reboots">5. Make it survive reboots<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#5-make-it-survive-reboots" class="hash-link" aria-label="Direct link to 5. Make it survive reboots" title="Direct link to 5. Make it survive reboots" translate="no">​</a></h3>
<p>Stop the current stack before setting up systemd:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">docker compose down</span><br></span></code></pre></div></div>
<p>Then create a systemd unit:</p>
<div class="language-ini codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-ini codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain"># /etc/systemd/system/myapp.service</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Unit]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Description=My Docker Compose App</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">After=network.target</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Service]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">Type=oneshot</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">WorkingDirectory=/root/demo-django</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">ExecStart=/usr/bin/docker compose up -d</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">ExecStop=/usr/bin/docker compose down</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">RemainAfterExit=yes</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain" style="display:inline-block"></span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">[Install]</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">WantedBy=multi-user.target</span><br></span></code></pre></div></div>
<p>Enable and start it:</p>
<div class="language-bash codeBlockContainer_Ckt0 theme-code-block" style="--prism-color:#393A34;--prism-background-color:#f6f8fa"><div class="codeBlockContent_QJqH"><pre tabindex="0" class="prism-code language-bash codeBlock_bY9V thin-scrollbar" style="color:#393A34;background-color:#f6f8fa"><code class="codeBlockLines_e6Vv"><span class="token-line" style="color:#393A34"><span class="token plain">systemctl enable myapp</span><br></span><span class="token-line" style="color:#393A34"><span class="token plain">systemctl start myapp</span><br></span></code></pre></div></div>
<p>Your app will now automatically start after reboots.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="6-handle-volumes-backups-logs">6. Handle volumes, backups, logs…<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#6-handle-volumes-backups-logs" class="hash-link" aria-label="Direct link to 6. Handle volumes, backups, logs…" title="Direct link to 6. Handle volumes, backups, logs…" translate="no">​</a></h3>
<p>If your Compose file uses volumes (e.g. MySQL, Redis), you now have to:</p>
<ul>
<li class="">Make sure volume paths are persisted and backed up</li>
<li class="">Inspect logs manually or add a logging layer (e.g. Loki or Graylog)</li>
<li class="">Possibly add monitoring for CPU, RAM, or disk usage</li>
</ul>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-its-a-lot">😮‍💨 It's a Lot<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#-its-a-lot" class="hash-link" aria-label="Direct link to 😮‍💨 It's a Lot" title="Direct link to 😮‍💨 It's a Lot" translate="no">​</a></h2>
<p>Even if you're comfortable with the CLI, this gets repetitive fast:</p>
<ul>
<li class="">Every project = new VPS</li>
<li class="">Manual nginx tweaks</li>
<li class="">No dashboard, no metrics</li>
<li class="">No easy way to share access</li>
</ul>
<p>After doing this too many times for clients, side projects, and demos, I decided to build something that just… <strong>does it for me</strong>.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-the-easy-way-paste--deploy">🧃 The Easy Way: Paste &amp; Deploy<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#-the-easy-way-paste--deploy" class="hash-link" aria-label="Direct link to 🧃 The Easy Way: Paste &amp; Deploy" title="Direct link to 🧃 The Easy Way: Paste &amp; Deploy" translate="no">​</a></h2>
<p>I'm building <a class="" href="https://hostim.dev/">Hostim.dev</a> – a developer-first platform that lets you paste your <code>docker-compose.yml</code>, click "Deploy", and you're live.</p>
<blockquote>
<p>Well, unless you <code>docker-compose.yml</code> is crazy big with tons of services, then it might take some manual configuration.</p>
</blockquote>
<p>Here's what it takes:</p>
<ol>
<li class="">Sign up (no credit card required)</li>
<li class="">Create a project and paste your Compose file</li>
<li class="">We generate your stack – apps, databases, volumes</li>
<li class="">Logs, metrics, and HTTPS just work</li>
</ol>
<div style="margin:2rem 0;background:#f9f9f9;border-radius:12px;box-shadow:0 4px 12px rgba(0, 0, 0, 0.06);padding:1rem"><video src="/videos/compose-cut-cut.mp4" controls="" playsinline="" muted="" style="width:100%;border-radius:8px"></video></div>
<p>No nginx. No SSH. No firewalls. Just a real app online.</p>
<hr>
<h2 class="anchor anchorTargetStickyNavbar_Vzrq" id="-try-it-free">🧪 Try It Free<a href="https://hostim.dev/blog/how-to-self-host-docker-compose/#-try-it-free" class="hash-link" aria-label="Direct link to 🧪 Try It Free" title="Direct link to 🧪 Try It Free" translate="no">​</a></h2>
<p>Every new user gets a 5-day trial project, plus <strong>always-free</strong> (albeit small) tiers for:</p>
<ul>
<li class="">MySQL and Postgres</li>
<li class="">Redis</li>
<li class="">Persistent volumes</li>
</ul>
<p>If you're tired of fighting servers and YAML, check it out:</p>
<p>👉 <a href="https://console.hostim.dev/" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_console">Get started with Hostim.dev</a></p>
<blockquote>
<p>🚀 Hostim.dev is currently in closed beta – if you want early access, <a class="" href="https://hostim.dev/">join the waitlist</a> or email me at <a href="mailto:pv@hostim.dev" target="_blank" rel="noopener noreferrer" class="">pv@hostim.dev</a></p>
</blockquote>
<hr>
<p><strong>P.S.</strong> If you <em>do</em> enjoy the ops side – I get it. I used to too. But these days, I just want to ship faster. That's what this is all about.</p>]]></content>
        <category label="docker" term="docker"/>
        <category label="docker-compose" term="docker-compose"/>
        <category label="selfhosting" term="selfhosting"/>
        <category label="devops" term="devops"/>
        <category label="tutorial" term="tutorial"/>
    </entry>
    <entry>
        <title type="html"><![CDATA[Why I'm Building Hostim.dev]]></title>
        <id>https://hostim.dev/blog/why-i-built-hostim/</id>
        <link href="https://hostim.dev/blog/why-i-built-hostim/"/>
        <updated>2025-07-28T00:00:00.000Z</updated>
        <summary type="html"><![CDATA[Hostim.dev is a solo-engineer project to bring hosting back to earth – fast full-stack app deployments without cloud rent or VC bloat.]]></summary>
        <content type="html"><![CDATA[<p>These days, hosting your app often means choosing between complexity, lock-in, and sky-high pricing. Whether it's a shiny new platform or a slick developer tool, most of them are just wrappers around the same old giants: <strong>AWS</strong>, <strong>GCP</strong>, and <strong>Azure</strong>.</p>
<p>And those giants are <strong>expensive by design</strong>.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-the-real-cost-of-hosting">💸 The Real Cost of Hosting<a href="https://hostim.dev/blog/why-i-built-hostim/#-the-real-cost-of-hosting" class="hash-link" aria-label="Direct link to 💸 The Real Cost of Hosting" title="Direct link to 💸 The Real Cost of Hosting" translate="no">​</a></h3>
<p>Take this comparison:</p>
<ul>
<li class=""><strong>Hetzner</strong>: 14 cores, 64 GB RAM → <strong>€52/month</strong>, cancel anytime</li>
<li class=""><strong>AWS r6g.2xlarge</strong> (8 vCPU, 64 GB RAM):<!-- -->
<ul>
<li class=""><strong>$133/month</strong> with 3-year reservation, upfront</li>
<li class=""><strong>$355/month</strong> on-demand</li>
</ul>
</li>
</ul>
<p>That's <strong>7× more expensive</strong> – for <strong>less power</strong> – and that's before you pay for:</p>
<ul>
<li class="">Egress traffic</li>
<li class="">Storage</li>
<li class="">Managed services</li>
<li class="">"Hidden" costs like snapshots, logs, and bandwidth</li>
</ul>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-and-thats-just-the-first-layer">🧃 And That's Just the First Layer<a href="https://hostim.dev/blog/why-i-built-hostim/#-and-thats-just-the-first-layer" class="hash-link" aria-label="Direct link to 🧃 And That's Just the First Layer" title="Direct link to 🧃 And That's Just the First Layer" translate="no">​</a></h3>
<p>Now stack on the second layer: the cool-looking hosting platform <em>you</em> chose.</p>
<p>They're backed by VC. They're growing fast.
And guess who's funding their founders, engineers, and investors?</p>
<p>You are.</p>
<p>If you're paying 7× more on AWS, and then 2× more through a middleman, that's not convenience – that's <strong>Cloud Rent</strong>.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-why-im-building-hostimdev">🌱 Why I'm Building Hostim.dev<a href="https://hostim.dev/blog/why-i-built-hostim/#-why-im-building-hostimdev" class="hash-link" aria-label="Direct link to 🌱 Why I'm Building Hostim.dev" title="Direct link to 🌱 Why I'm Building Hostim.dev" translate="no">​</a></h3>
<p>Hostim.dev is my answer to all of this.
It's a <strong>bare-metal, developer-first PaaS</strong> that puts fairness and simplicity first.</p>
<p>I'm a DevOps engineer building this solo. No VC. No team. Just a focused mission:</p>
<blockquote>
<p>🛠 <strong>Let anyone deploy full-stack apps at fair prices – without big cloud bloat.</strong></p>
</blockquote>
<p><strong>Here's what you get:</strong></p>
<ul>
<li class="">Deploy from <strong>Docker</strong>, <strong>Git</strong>, or <strong>Docker Compose</strong></li>
<li class="">Built-in <strong>PostgreSQL, MySQL, Redis</strong>, and <strong>Volumes</strong></li>
<li class="">Real-time <strong>logs</strong>, <strong>metrics</strong>, and <strong>auto HTTPS</strong></li>
<li class=""><strong>Per-project isolation</strong> with internal networking</li>
<li class="">A <strong>5-day free trial</strong> for any project</li>
<li class="">Always-free <strong>tiers</strong> for databases, Redis, and volumes – perfect for dev and pet projects</li>
</ul>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-where-were-starting">🗺 Where We're Starting<a href="https://hostim.dev/blog/why-i-built-hostim/#-where-were-starting" class="hash-link" aria-label="Direct link to 🗺 Where We're Starting" title="Direct link to 🗺 Where We're Starting" translate="no">​</a></h3>
<ul>
<li class="">Our first region is <strong>Germany-based</strong></li>
<li class="">A <strong>US rollout is planned</strong></li>
</ul>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-what-its-for-and-not-for">🧩 What It's For (and Not For)<a href="https://hostim.dev/blog/why-i-built-hostim/#-what-its-for-and-not-for" class="hash-link" aria-label="Direct link to 🧩 What It's For (and Not For)" title="Direct link to 🧩 What It's For (and Not For)" translate="no">​</a></h3>
<p>If it fits in a Docker container, you can host it on Hostim.dev.</p>
<p>Right now it's built for <strong>web apps</strong> – dashboards, APIs, sites, admin panels, AI demos, side projects, SaaS backends. But we're flexible and evolving fast based on user feedback.</p>
<p>We're not trying to out-feature big players. We're trying to <strong>strip away what you don't need</strong> and make what you do need <strong>accessible</strong>.</p>
<hr>
<h3 class="anchor anchorTargetStickyNavbar_Vzrq" id="-whats-next">🚀 What's Next<a href="https://hostim.dev/blog/why-i-built-hostim/#-whats-next" class="hash-link" aria-label="Direct link to 🚀 What's Next" title="Direct link to 🚀 What's Next" translate="no">​</a></h3>
<p>We've launched! You can try Hostim.dev right now, no sign-up required.</p>
<ul>
<li class="">Try it out: <a href="https://console.hostim.dev/dashboard?preview=1&amp;modal=1" target="_blank" rel="noopener noreferrer" data-umami-event="blog_click_try_now"><b>hostim.dev</b></a></li>
<li class="">Browse the <a class="" href="https://hostim.dev/docs/getting-started/">docs</a></li>
</ul>
<p>Let's stop overpaying for complexity.
Let's bring hosting back to earth.</p>]]></content>
        <category label="self-hosting" term="self-hosting"/>
        <category label="paas" term="paas"/>
        <category label="devops" term="devops"/>
        <category label="startup" term="startup"/>
    </entry>
</feed>