<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: Rizky Bachtiar Irwanto</title>
    <description>The latest articles on Forem by Rizky Bachtiar Irwanto (@ribato).</description>
    <link>https://forem.com/ribato</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3774744%2Ff3a3e459-0945-4cbc-b133-6a5d3395550b.png</url>
      <title>Forem: Rizky Bachtiar Irwanto</title>
      <link>https://forem.com/ribato</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/ribato"/>
    <language>en</language>
    <item>
      <title>Building MultiWA: An Open-Source Self-Hosted WhatsApp API Gateway</title>
      <dc:creator>Rizky Bachtiar Irwanto</dc:creator>
      <pubDate>Mon, 16 Feb 2026 02:34:00 +0000</pubDate>
      <link>https://forem.com/ribato/building-multiwa-an-open-source-self-hosted-whatsapp-api-gateway-2me1</link>
      <guid>https://forem.com/ribato/building-multiwa-an-open-source-self-hosted-whatsapp-api-gateway-2me1</guid>
      <description>&lt;h2&gt;
  
  
  Why I Built MultiWA
&lt;/h2&gt;

&lt;p&gt;I needed WhatsApp automation for a business application, but the official WhatsApp Cloud API comes with per-message pricing and requires Meta hosting. Existing open-source alternatives were either limited to a single session or lacked proper admin tooling.&lt;/p&gt;

&lt;p&gt;So I built &lt;strong&gt;MultiWA&lt;/strong&gt; — a fully self-hosted, multi-session WhatsApp API gateway with an admin dashboard, visual automation builder, and AI-powered auto-replies.&lt;/p&gt;

&lt;h2&gt;
  
  
  What is MultiWA?
&lt;/h2&gt;

&lt;p&gt;MultiWA is an open-source WhatsApp Business API Gateway that lets you:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Connect &lt;strong&gt;multiple WhatsApp numbers&lt;/strong&gt; through a single unified API&lt;/li&gt;
&lt;li&gt;Choose between &lt;strong&gt;whatsapp-web.js&lt;/strong&gt; and &lt;strong&gt;Baileys&lt;/strong&gt; engine adapters&lt;/li&gt;
&lt;li&gt;Manage everything through a modern &lt;strong&gt;Admin Dashboard&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;Build automations with a &lt;strong&gt;visual drag-and-drop flow builder&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;Set up &lt;strong&gt;AI-powered auto-replies&lt;/strong&gt; using OpenAI or Google AI&lt;/li&gt;
&lt;li&gt;Send messages via &lt;strong&gt;REST API&lt;/strong&gt; with official SDKs for TypeScript, Python, and PHP&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Architecture
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;┌──────────────────────────────────────────────┐
│               Nginx (SSL/Proxy)               │
├──────────────────┬───────────────────────────┤
│  Admin (Next.js) │     API (NestJS/Fastify)   │
│  Port 3001       │     Port 3000              │
│                  │  WhatsApp Engine Adapters   │
│                  │  ├─ whatsapp-web.js         │
│  Worker (BullMQ) │  └─ Baileys                │
│                  │  PostgreSQL │ Redis         │
└──────────────────┴───────────────────────────┘
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  Tech Stack
&lt;/h3&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Layer&lt;/th&gt;
&lt;th&gt;Technology&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;API&lt;/td&gt;
&lt;td&gt;NestJS 10 + Fastify&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Admin&lt;/td&gt;
&lt;td&gt;Next.js 14 + Tailwind CSS&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Database&lt;/td&gt;
&lt;td&gt;PostgreSQL 16 + Prisma ORM&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Queue&lt;/td&gt;
&lt;td&gt;Redis 7 + BullMQ&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;WhatsApp&lt;/td&gt;
&lt;td&gt;whatsapp-web.js / Baileys&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Auth&lt;/td&gt;
&lt;td&gt;JWT (access + refresh tokens)&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Realtime&lt;/td&gt;
&lt;td&gt;Socket.IO&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;Container&lt;/td&gt;
&lt;td&gt;Docker + Docker Compose&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;h2&gt;
  
  
  Key Features
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Multi-Session Management
&lt;/h3&gt;

&lt;p&gt;Connect unlimited WhatsApp accounts, each running independently with its own QR code authentication, status monitoring, and message routing.&lt;/p&gt;

&lt;h3&gt;
  
  
  Pluggable Engine Adapters
&lt;/h3&gt;

&lt;p&gt;MultiWA uses an adapter pattern for WhatsApp engines. Currently supports whatsapp-web.js (Chromium-based) and Baileys (lightweight, no browser). You can switch engines per session without changing your application code.&lt;/p&gt;

&lt;h3&gt;
  
  
  Visual Automation Builder
&lt;/h3&gt;

&lt;p&gt;Create message automation flows with a drag-and-drop interface. Define triggers (keyword match, regex, webhook), conditions, and actions (reply, forward, API call) — all without writing code.&lt;/p&gt;

&lt;h3&gt;
  
  
  AI Knowledge Base
&lt;/h3&gt;

&lt;p&gt;Upload documents and let AI handle customer inquiries automatically. Supports both OpenAI and Google AI as providers. The system uses RAG (Retrieval-Augmented Generation) to find relevant answers from your knowledge base.&lt;/p&gt;

&lt;h3&gt;
  
  
  Enterprise Features
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;JWT authentication with refresh tokens&lt;/li&gt;
&lt;li&gt;API key management with scoping and expiration&lt;/li&gt;
&lt;li&gt;Webhook subscriptions for real-time event notifications&lt;/li&gt;
&lt;li&gt;Scheduled messages and broadcast&lt;/li&gt;
&lt;li&gt;Audit trail and analytics dashboard&lt;/li&gt;
&lt;li&gt;GDPR-compliant data export and deletion&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Quick Start with Docker
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;git clone https://github.com/ribato22/MultiWA.git
&lt;span class="nb"&gt;cd &lt;/span&gt;MultiWA
&lt;span class="nb"&gt;cp&lt;/span&gt; .env.production.example .env
docker compose &lt;span class="nt"&gt;-f&lt;/span&gt; docker-compose.production.yml up &lt;span class="nt"&gt;-d&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's it. API runs on port 3333 with Swagger docs, Admin dashboard on port 3001.&lt;/p&gt;

&lt;h2&gt;
  
  
  Send Your First Message
&lt;/h2&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;curl &lt;span class="nt"&gt;-X&lt;/span&gt; POST http://localhost:3333/api/v1/messages/send &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;-H&lt;/span&gt; &lt;span class="s2"&gt;"Authorization: Bearer YOUR_TOKEN"&lt;/span&gt; &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;-H&lt;/span&gt; &lt;span class="s2"&gt;"Content-Type: application/json"&lt;/span&gt; &lt;span class="se"&gt;\&lt;/span&gt;
  &lt;span class="nt"&gt;-d&lt;/span&gt; &lt;span class="s1"&gt;'{
    "profileId": "profile-uuid",
    "to": "6281234567890",
    "message": "Hello from MultiWA!"
  }'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  SDK Support
&lt;/h2&gt;

&lt;p&gt;Official SDKs are available for three languages:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;MultiWA&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;@multiwa/sdk&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;MultiWA&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;baseUrl&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;http://localhost:3000&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;apiKey&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;your-api-key&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;});&lt;/span&gt;

&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;client&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;messages&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;send&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;profileId&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;profile-uuid&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;to&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;6281234567890&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="na"&gt;message&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;Hello!&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;});&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Contributing
&lt;/h2&gt;

&lt;p&gt;MultiWA is MIT-licensed and open for contributions. Whether it's bug reports, feature requests, or pull requests — everything is welcome.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;GitHub&lt;/strong&gt;: &lt;a href="https://github.com/ribato22/MultiWA" rel="noopener noreferrer"&gt;https://github.com/ribato22/MultiWA&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;If you have any questions about the architecture or need help setting it up, feel free to ask in the comments or open a discussion on GitHub.&lt;/p&gt;

</description>
      <category>opensource</category>
      <category>webdev</category>
      <category>docker</category>
      <category>typescript</category>
    </item>
  </channel>
</rss>
