<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Microservices on TECHFOR by Suriya Sonphu</title><link>http://suriyasonphu.com/tags/microservices/</link><description>Recent content in Microservices on TECHFOR by Suriya Sonphu</description><generator>Hugo -- gohugo.io</generator><language>th</language><lastBuildDate>Sun, 27 Jul 2025 00:00:00 +0000</lastBuildDate><atom:link href="http://suriyasonphu.com/tags/microservices/index.xml" rel="self" type="application/rss+xml"/><item><title>Modular Monolith: สถาปัตยกรรมที่ผสมผสานข้อดีของ Monolith และ Microservices</title><link>http://suriyasonphu.com/post/2025-07-27-modular-monolith-architecture/</link><pubDate>Sun, 27 Jul 2025 00:00:00 +0000</pubDate><guid>http://suriyasonphu.com/post/2025-07-27-modular-monolith-architecture/</guid><description>&lt;img src="http://suriyasonphu.com/post/2025-07-27-modular-monolith-architecture/modular-monolith-cover.png" alt="Featured image of post Modular Monolith: สถาปัตยกรรมที่ผสมผสานข้อดีของ Monolith และ Microservices" />&lt;h2 id="modular-monolith-คออะไร-และทำไมถงเปนทางเลอกทนาสนใจ">Modular Monolith คืออะไร และทำไมถึงเป็นทางเลือกที่น่าสนใจ
&lt;/h2>&lt;p>Modular Monolith เป็นสถาปัตยกรรมซอฟต์แวร์ที่ผสมผสานข้อดีของ &lt;strong>Monolith&lt;/strong> และ &lt;strong>Microservices&lt;/strong> เข้าด้วยกัน โดยการจัดระเบียบโค้ดให้เป็นโมดูลที่มีขอบเขตชัดเจน (bounded context) แต่ยังคงการ deploy เป็นหน่วยเดียวกัน&lt;/p>
&lt;h3 id="ปญหาของ-traditional-monolith">ปัญหาของ Traditional Monolith
&lt;/h3>&lt;p>&lt;strong>Traditional Monolith&lt;/strong> มักจะประสบปัญหา:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Tightly Coupled Code&lt;/strong>: โค้ดทุกส่วนผูกติดกันแน่น ยากต่อการแก้ไข&lt;/li>
&lt;li>&lt;strong>Single Point of Failure&lt;/strong>: หากส่วนใดส่วนหนึ่งเสีย ระบบทั้งหมดล้มเหลว&lt;/li>
&lt;li>&lt;strong>Technology Lock-in&lt;/strong>: ต้องใช้เทคโนโลยีเดียวกันทั้งแอปพลิเคชัน&lt;/li>
&lt;li>&lt;strong>Difficult to Scale&lt;/strong>: ไม่สามารถปรับขนาดแค่ส่วนที่ต้องการได้&lt;/li>
&lt;/ul>
&lt;h3 id="ปญหาของ-microservices">ปัญหาของ Microservices
&lt;/h3>&lt;p>แม้ &lt;strong>Microservices&lt;/strong> จะแก้ปัญหาหลายอย่างของ Monolith แต่ก็มีความซับซ้อนเพิ่มขึ้น:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Distributed System Complexity&lt;/strong>: ความซับซ้อนของระบบกระจาย&lt;/li>
&lt;li>&lt;strong>Network Latency&lt;/strong>: ความล่าช้าจากการสื่อสารผ่าน network&lt;/li>
&lt;li>&lt;strong>Data Consistency&lt;/strong>: ปัญหาความสอดคล้องของข้อมูล&lt;/li>
&lt;li>&lt;strong>Operational Overhead&lt;/strong>: ต้องการทีม DevOps ที่แข็งแกร่ง&lt;/li>
&lt;li>&lt;strong>Development Complexity&lt;/strong>: ความซับซ้อนในการพัฒนาและ debug&lt;/li>
&lt;/ul>
&lt;h3 id="modular-monolith-ทางออกทสมดล">Modular Monolith: ทางออกที่สมดุล
&lt;/h3>&lt;p>Modular Monolith นำเสนอวิธีการที่สมดุล:&lt;/p>
&lt;p>&lt;strong>✅ ข้อดีที่ได้รับ:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Modularity&lt;/strong>: การแบ่งโมดูลที่ชัดเจนตาม business domain&lt;/li>
&lt;li>&lt;strong>Loose Coupling&lt;/strong>: โมดูลสื่อสารผ่าน interfaces ที่กำหนดไว้&lt;/li>
&lt;li>&lt;strong>Single Deployment&lt;/strong>: ง่ายต่อการ deploy และ maintain&lt;/li>
&lt;li>&lt;strong>Easier Testing&lt;/strong>: ทดสอบได้ง่ายกว่า distributed systems&lt;/li>
&lt;li>&lt;strong>Gradual Migration&lt;/strong>: สามารถแยกเป็น microservices ได้ในอนาคต&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>⚠️ ข้อควรพิจารณา:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Technology Constraint&lt;/strong>: ยังคงถูกจำกัดเทคโนโลยีหลัก&lt;/li>
&lt;li>&lt;strong>Shared Database&lt;/strong>: อาจมีปัญหา data coupling&lt;/li>
&lt;li>&lt;strong>Resource Scaling&lt;/strong>: ไม่สามารถ scale แต่ละโมดูลแยกได้&lt;/li>
&lt;/ul>
&lt;h2 id="โครงสรางของ-modular-monolith-e-commerce-application">โครงสร้างของ Modular Monolith E-commerce Application
&lt;/h2>&lt;p>ผมได้สร้างตัวอย่าง &lt;a class="link" href="https://github.com/suriyasonp/modular-monolith-ecommerce" target="_blank" rel="noopener"
>Modular Monolith E-commerce Application&lt;/a> ด้วย C# และ ASP.NET Core เพื่อแสดงให้เห็นการประยุกต์ใช้หลักการนี้ในการพัฒนาระบบจริง&lt;/p>
&lt;h3 id="-architecture-overview">🏗️ Architecture Overview
&lt;/h3>&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-text" data-lang="text">&lt;span style="display:flex;">&lt;span>ECommerceApp/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>├── src/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── ECommerceApp/ # Main Web API Application
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Controllers/ # API Controllers
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Program.cs # Application entry point
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ └── ECommerceApp.csproj
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── ECommerceApp.Shared/ # Shared Kernel
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Events/ # Domain Events
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Kernel/ # Base entities, interfaces
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ └── ECommerceApp.Shared.csproj
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ └── Modules/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── Orders/ # Orders Module
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Application/ # Use cases, commands, queries
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Domain/ # Domain entities, events
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Infrastructure/ # Data access, repositories
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── OrdersModule.cs # Module registration
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ └── ECommerceApp.Modules.Orders.csproj
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── Products/ # Products Module
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Application/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Domain/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── Infrastructure/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ ├── ProductsModule.cs
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ │ └── ECommerceApp.Modules.Products.csproj
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ └── Customers/ # Customers Module
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── Domain/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── Infrastructure/
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ ├── CustomersModule.cs
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>│ └── ECommerceApp.Modules.Customers.csproj
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="-หลกการสำคญของ-modular-monolith">🎯 หลักการสำคัญของ Modular Monolith
&lt;/h3>&lt;p>&lt;strong>1. Modules (โมดูล)&lt;/strong>
แต่ละโมดูลแทนความสามารถทางธุรกิจที่แตกต่างกัน เช่น Orders, Products, Customers&lt;/p>
&lt;p>&lt;strong>2. Encapsulation (การห่อหุ้ม)&lt;/strong>
โมดูลซ่อนรายละเอียดภายในและเปิดเผยเฉพาะ functionality ที่จำเป็นผ่าน interfaces&lt;/p>
&lt;p>&lt;strong>3. Shared Kernel (แกนร่วม)&lt;/strong>
ส่วนที่ใช้ร่วมกัน เช่น base entities, common utilities, domain events&lt;/p>
&lt;p>&lt;strong>4. Communication (การสื่อสาร)&lt;/strong>
โมดูลสื่อสารผ่าน interfaces หรือ domain events หลีกเลี่ยงการพึ่งพาโดยตรง&lt;/p>
&lt;p>&lt;strong>5. Single Deployment (การ Deploy เดียว)&lt;/strong>
โมดูลทั้งหมดถูก deploy ร่วมกันเป็นหน่วยเดียว&lt;/p>
&lt;h2 id="การออกแบบโมดลตาม-domain-driven-design">การออกแบบโมดูลตาม Domain-Driven Design
&lt;/h2>&lt;h3 id="products-module">Products Module
&lt;/h3>&lt;p>&lt;strong>Domain Layer:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">Product&lt;/span> : BaseEntity
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">string&lt;/span> Name { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">string&lt;/span> Description { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">decimal&lt;/span> Price { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">int&lt;/span> StockQuantity { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> UpdatePrice(&lt;span style="color:#66d9ef">decimal&lt;/span> newPrice)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">if&lt;/span> (newPrice &amp;lt;= &lt;span style="color:#ae81ff">0&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">throw&lt;/span> &lt;span style="color:#66d9ef">new&lt;/span> ArgumentException(&lt;span style="color:#e6db74">&amp;#34;Price must be positive&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Price = newPrice;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Raise domain event&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RaiseDomainEvent(&lt;span style="color:#66d9ef">new&lt;/span> ProductPriceUpdatedEvent(Id, newPrice));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> UpdateStock(&lt;span style="color:#66d9ef">int&lt;/span> quantity)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">if&lt;/span> (StockQuantity + quantity &amp;lt; &lt;span style="color:#ae81ff">0&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">throw&lt;/span> &lt;span style="color:#66d9ef">new&lt;/span> InvalidOperationException(&lt;span style="color:#e6db74">&amp;#34;Insufficient stock&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> StockQuantity += quantity;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RaiseDomainEvent(&lt;span style="color:#66d9ef">new&lt;/span> ProductStockUpdatedEvent(Id, StockQuantity));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>Application Layer:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">UpdateProductPriceCommand&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> Guid ProductId { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">decimal&lt;/span> NewPrice { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">UpdateProductPriceHandler&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">readonly&lt;/span> IProductRepository _repository;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;Result&amp;gt; Handle(UpdateProductPriceCommand command)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> product = &lt;span style="color:#66d9ef">await&lt;/span> _repository.GetByIdAsync(command.ProductId);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">if&lt;/span> (product == &lt;span style="color:#66d9ef">null&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> Result.Failure(&lt;span style="color:#e6db74">&amp;#34;Product not found&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> product.UpdatePrice(command.NewPrice);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> _repository.UpdateAsync(product);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> Result.Success();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="orders-module">Orders Module
&lt;/h3>&lt;p>&lt;strong>Domain Layer:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">Order&lt;/span> : BaseEntity
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> Guid CustomerId { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> OrderStatus Status { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> List&amp;lt;OrderItem&amp;gt; Items { &lt;span style="color:#66d9ef">get&lt;/span>; &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">set&lt;/span>; } = &lt;span style="color:#66d9ef">new&lt;/span>();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">decimal&lt;/span> TotalAmount =&amp;gt; Items.Sum(item =&amp;gt; item.TotalPrice);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> Order Create(Guid customerId, List&amp;lt;OrderItem&amp;gt; items)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> order = &lt;span style="color:#66d9ef">new&lt;/span> Order
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> CustomerId = customerId,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Status = OrderStatus.Pending,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Items = items
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> };
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> order.RaiseDomainEvent(&lt;span style="color:#66d9ef">new&lt;/span> OrderCreatedEvent(order.Id, customerId));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> order;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> Confirm()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">if&lt;/span> (Status != OrderStatus.Pending)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">throw&lt;/span> &lt;span style="color:#66d9ef">new&lt;/span> InvalidOperationException(&lt;span style="color:#e6db74">&amp;#34;Only pending orders can be confirmed&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Status = OrderStatus.Confirmed;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RaiseDomainEvent(&lt;span style="color:#66d9ef">new&lt;/span> OrderConfirmedEvent(Id));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="การสอสารระหวางโมดล">การสื่อสารระหว่างโมดูล
&lt;/h3>&lt;p>&lt;strong>Interface-based Communication:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// ใน Orders Module&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">interface&lt;/span> &lt;span style="color:#a6e22e">IProductService&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Task&amp;lt;Product?&amp;gt; GetProductAsync(Guid productId);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Task&amp;lt;&lt;span style="color:#66d9ef">bool&lt;/span>&amp;gt; IsProductAvailableAsync(Guid productId, &lt;span style="color:#66d9ef">int&lt;/span> quantity);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// ใน Products Module&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">ProductService&lt;/span> : IProductService
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">readonly&lt;/span> IProductRepository _repository;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;Product?&amp;gt; GetProductAsync(Guid productId)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> &lt;span style="color:#66d9ef">await&lt;/span> _repository.GetByIdAsync(productId);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task&amp;lt;&lt;span style="color:#66d9ef">bool&lt;/span>&amp;gt; IsProductAvailableAsync(Guid productId, &lt;span style="color:#66d9ef">int&lt;/span> quantity)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> product = &lt;span style="color:#66d9ef">await&lt;/span> _repository.GetByIdAsync(productId);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> product != &lt;span style="color:#66d9ef">null&lt;/span> &amp;amp;&amp;amp; product.StockQuantity &amp;gt;= quantity;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>Event-based Communication (สำหรับอนาคต):&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderConfirmedEvent&lt;/span> : IDomainEvent
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> Guid OrderId { &lt;span style="color:#66d9ef">get&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> DateTime OccurredOn { &lt;span style="color:#66d9ef">get&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> OrderConfirmedEvent(Guid orderId)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> OrderId = orderId;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> OccurredOn = DateTime.UtcNow;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Event Handler ใน Products Module&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderConfirmedEventHandler&lt;/span> : IEventHandler&amp;lt;OrderConfirmedEvent&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task Handle(OrderConfirmedEvent @event)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Update inventory, send notifications, etc.&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h2 id="การตดตงและรนแอปพลเคชน">การติดตั้งและรันแอปพลิเคชัน
&lt;/h2>&lt;h3 id="prerequisites">Prerequisites
&lt;/h3>&lt;ul>
&lt;li>.NET 8.0 SDK&lt;/li>
&lt;li>Visual Studio Code หรือ Visual Studio&lt;/li>
&lt;/ul>
&lt;h3 id="การเรมตนใชงาน">การเริ่มต้นใช้งาน
&lt;/h3>&lt;p>&lt;strong>1. Clone Repository:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-bash" data-lang="bash">&lt;span style="display:flex;">&lt;span>git clone https://github.com/suriyasonp/modular-monolith-ecommerce.git
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>cd modular-monolith-ecommerce
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. รันแอปพลิเคชัน:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-bash" data-lang="bash">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e"># วิธีที่ 1: ใช้ script (แนะนำ)&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>./run.sh
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e"># วิธีที่ 2: รันด้วยตนเอง&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>dotnet build
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>dotnet run --project src/ECommerceApp/ECommerceApp.csproj
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>3. เข้าถึง API:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Swagger UI&lt;/strong>: &lt;a class="link" href="http://localhost:5000/swagger/index.html" target="_blank" rel="noopener"
>http://localhost:5000/swagger/index.html&lt;/a>&lt;/li>
&lt;li>&lt;strong>API Base URL&lt;/strong>: &lt;a class="link" href="http://localhost:5000/api" target="_blank" rel="noopener"
>http://localhost:5000/api&lt;/a>&lt;/li>
&lt;li>&lt;strong>Sample Products&lt;/strong>: ข้อมูลจะถูกสร้างอัตโนมัติเมื่อเริ่มระบบ&lt;/li>
&lt;/ul>
&lt;h3 id="การทดสอบ-api">การทดสอบ API
&lt;/h3>&lt;p>&lt;strong>Quick Test Script:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-bash" data-lang="bash">&lt;span style="display:flex;">&lt;span>./test-api.sh
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>การทดสอบด้วยตนเอง:&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-bash" data-lang="bash">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e"># ดู Products ทั้งหมด&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>curl http://localhost:5000/api/products
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e"># สร้าง Customer ใหม่&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>curl -X POST http://localhost:5000/api/customers &lt;span style="color:#ae81ff">\
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#ae81ff">&lt;/span> -H &lt;span style="color:#e6db74">&amp;#34;Content-Type: application/json&amp;#34;&lt;/span> &lt;span style="color:#ae81ff">\
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#ae81ff">&lt;/span> -d &lt;span style="color:#e6db74">&amp;#39;{
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;firstName&amp;#34;: &amp;#34;สมชาย&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;lastName&amp;#34;: &amp;#34;ใจดี&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;email&amp;#34;: &amp;#34;somchai@example.com&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;address&amp;#34;: {
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;street&amp;#34;: &amp;#34;123 ถนนสุขุมวิท&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;city&amp;#34;: &amp;#34;กรุงเทพฯ&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;state&amp;#34;: &amp;#34;กทม&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;zipCode&amp;#34;: &amp;#34;10110&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;country&amp;#34;: &amp;#34;ไทย&amp;#34;
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> }
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> }&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e"># สร้าง Order ใหม่&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>curl -X POST http://localhost:5000/api/orders &lt;span style="color:#ae81ff">\
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#ae81ff">&lt;/span> -H &lt;span style="color:#e6db74">&amp;#34;Content-Type: application/json&amp;#34;&lt;/span> &lt;span style="color:#ae81ff">\
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#ae81ff">&lt;/span> -d &lt;span style="color:#e6db74">&amp;#39;{
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;customerId&amp;#34;: &amp;#34;customer-guid-here&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;items&amp;#34;: [
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> {
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;productId&amp;#34;: &amp;#34;product-guid-here&amp;#34;,
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> &amp;#34;quantity&amp;#34;: 2
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> }
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> ]
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#e6db74"> }&amp;#39;&lt;/span>
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h2 id="api-endpoints-ทสำคญ">API Endpoints ที่สำคัญ
&lt;/h2>&lt;h3 id="products-api">Products API
&lt;/h3>&lt;ul>
&lt;li>&lt;code>GET /api/products&lt;/code> - ดูสินค้าทั้งหมด&lt;/li>
&lt;li>&lt;code>GET /api/products/{id}&lt;/code> - ดูสินค้าตาม ID&lt;/li>
&lt;li>&lt;code>POST /api/products&lt;/code> - เพิ่มสินค้าใหม่&lt;/li>
&lt;li>&lt;code>PUT /api/products/{id}/price&lt;/code> - อัปเดตราคาสินค้า&lt;/li>
&lt;li>&lt;code>PUT /api/products/{id}/stock&lt;/code> - อัปเดตสต็อกสินค้า&lt;/li>
&lt;/ul>
&lt;h3 id="customers-api">Customers API
&lt;/h3>&lt;ul>
&lt;li>&lt;code>GET /api/customers&lt;/code> - ดูลูกค้าทั้งหมด&lt;/li>
&lt;li>&lt;code>GET /api/customers/{id}&lt;/code> - ดูลูกค้าตาม ID&lt;/li>
&lt;li>&lt;code>POST /api/customers&lt;/code> - เพิ่มลูกค้าใหม่&lt;/li>
&lt;/ul>
&lt;h3 id="orders-api">Orders API
&lt;/h3>&lt;ul>
&lt;li>&lt;code>GET /api/orders&lt;/code> - ดูออเดอร์ทั้งหมด&lt;/li>
&lt;li>&lt;code>GET /api/orders/{id}&lt;/code> - ดูออเดอร์ตาม ID&lt;/li>
&lt;li>&lt;code>GET /api/orders/customer/{customerId}&lt;/code> - ดูออเดอร์ของลูกค้า&lt;/li>
&lt;li>&lt;code>POST /api/orders&lt;/code> - สร้างออเดอร์ใหม่&lt;/li>
&lt;li>&lt;code>POST /api/orders/{id}/confirm&lt;/code> - ยืนยันออเดอร์&lt;/li>
&lt;li>&lt;code>POST /api/orders/{id}/ship&lt;/code> - จัดส่งออเดอร์&lt;/li>
&lt;li>&lt;code>POST /api/orders/{id}/deliver&lt;/code> - ส่งมอบออเดอร์&lt;/li>
&lt;/ul>
&lt;h2 id="ขอดและขอเสยของ-modular-monolith">ข้อดีและข้อเสียของ Modular Monolith
&lt;/h2>&lt;h3 id="-ขอด">✅ ข้อดี
&lt;/h3>&lt;p>&lt;strong>1. ความเรียบง่ายในการ Deploy&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>Deploy เป็นหน่วยเดียว ลดความซับซ้อนของ infrastructure&lt;/li>
&lt;li>ไม่ต้องกังวลเรื่อง service discovery หรือ load balancing&lt;/li>
&lt;li>การ rollback ทำได้ง่าย&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>2. ประสิทธิภาพ&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ไม่มี network latency ระหว่างโมดูล&lt;/li>
&lt;li>การสื่อสารเป็น in-process calls&lt;/li>
&lt;li>Transaction ข้าม modules ทำได้ง่าย&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>3. การพัฒนาและ Debug&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>Debug ได้ง่ายกว่า distributed systems&lt;/li>
&lt;li>IDE support ดีกว่า&lt;/li>
&lt;li>End-to-end testing ง่ายกว่า&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>4. ต้นทุนการดำเนินงาน&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>Infrastructure requirements ต่ำกว่า&lt;/li>
&lt;li>ไม่ต้องมีทีม DevOps ขนาดใหญ่&lt;/li>
&lt;li>Monitoring และ logging ง่ายกว่า&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>5. ความยืดหยุ่นในอนาคต&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>สามารถแยกเป็น microservices ได้เมื่อจำเป็น&lt;/li>
&lt;li>โมดูลที่ออกแบบดีสามารถ extract ออกมาได้ง่าย&lt;/li>
&lt;/ul>
&lt;h3 id="-ขอเสย">⚠️ ข้อเสีย
&lt;/h3>&lt;p>&lt;strong>1. ข้อจำกัดด้านเทคโนโลยี&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ต้องใช้เทคโนโลยีหลักเดียวกัน&lt;/li>
&lt;li>ยากต่อการใช้ different tech stacks&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>2. การ Scale&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ไม่สามารถ scale แต่ละโมดูลแยกได้&lt;/li>
&lt;li>Resource utilization อาจไม่เหมาะสม&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>3. Database Coupling&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>หากใช้ shared database อาจมีปัญหา coupling&lt;/li>
&lt;li>Schema migration ต้องระวังผลกระทบ&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>4. Team Independence&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ทีมยังคงต้อง coordinate ใน deployment&lt;/li>
&lt;li>Shared codebase อาจเป็นปัญหาในทีมใหญ่&lt;/li>
&lt;/ul>
&lt;h2 id="เมอไหรควรใช-modular-monolith">เมื่อไหร่ควรใช้ Modular Monolith
&lt;/h2>&lt;h3 id="-เหมาะสำหรบ">🎯 เหมาะสำหรับ:
&lt;/h3>&lt;p>&lt;strong>1. ทีมขนาดเล็กถึงกลาง (2-20 คน)&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>มีทรัพยากรจำกัดสำหรับ infrastructure complexity&lt;/li>
&lt;li>ต้องการ development velocity สูง&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>2. แอปพลิเคชันใหม่&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ยังไม่แน่ใจเรื่อง domain boundaries&lt;/li>
&lt;li>ต้องการ rapid prototyping และ iteration&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>3. Requirements ยังไม่ชัดเจน&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>Business requirements เปลี่ยนแปลงบ่อย&lt;/li>
&lt;li>ต้องการความยืดหยุ่นในการปรับเปลี่ยน&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>4. Limited DevOps Capability&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ทีมยังไม่พร้อมสำหรับ distributed systems&lt;/li>
&lt;li>Infrastructure automation ยังไม่เข้มแข็ง&lt;/li>
&lt;/ul>
&lt;h3 id="-ไมเหมาะสำหรบ">🚫 ไม่เหมาะสำหรับ:
&lt;/h3>&lt;p>&lt;strong>1. องค์กรขนาดใหญ่ที่มีทีมมาก&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ทีมต้องการ independence สูง&lt;/li>
&lt;li>มี different release cycles&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>2. Requirements ที่แตกต่างกันมาก&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>แต่ละส่วนมี scalability requirements ต่างกัน&lt;/li>
&lt;li>ต้องการ different technologies&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>3. Compliance และ Security ข้อจำกัดสูง&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ต้องการ isolation ระหว่าง components&lt;/li>
&lt;li>มี regulatory requirements ที่เข้มงวด&lt;/li>
&lt;/ul>
&lt;h2 id="การวางแผน-migration-path">การวางแผน Migration Path
&lt;/h2>&lt;h3 id="phase-1-เรมตนดวย-modular-monolith">Phase 1: เริ่มต้นด้วย Modular Monolith
&lt;/h3>&lt;pre tabindex="0">&lt;code>Traditional Monolith → Modular Monolith
&lt;/code>&lt;/pre>&lt;p>&lt;strong>Activities:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ระบุ domain boundaries&lt;/li>
&lt;li>สร้าง modules ตาม business capabilities&lt;/li>
&lt;li>Implement interfaces สำหรับ inter-module communication&lt;/li>
&lt;li>เพิ่ม domain events infrastructure&lt;/li>
&lt;/ul>
&lt;h3 id="phase-2-ปรบปรง-module-independence">Phase 2: ปรับปรุง Module Independence
&lt;/h3>&lt;pre tabindex="0">&lt;code>Tightly Coupled Modules → Loosely Coupled Modules
&lt;/code>&lt;/pre>&lt;p>&lt;strong>Activities:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>แยก databases per module&lt;/li>
&lt;li>ใช้ event-driven communication&lt;/li>
&lt;li>Implement distributed tracing&lt;/li>
&lt;li>เพิ่ม monitoring per module&lt;/li>
&lt;/ul>
&lt;h3 id="phase-3-selective-microservices-extraction">Phase 3: Selective Microservices Extraction
&lt;/h3>&lt;pre tabindex="0">&lt;code>Modular Monolith → Hybrid Architecture
&lt;/code>&lt;/pre>&lt;p>&lt;strong>Activities:&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ระบุ modules ที่เหมาะสมสำหรับ extraction&lt;/li>
&lt;li>Extract modules ที่มี different scaling needs&lt;/li>
&lt;li>Implement API gateways&lt;/li>
&lt;li>Setup service mesh&lt;/li>
&lt;/ul>
&lt;h2 id="best-practices-สำหรบ-modular-monolith">Best Practices สำหรับ Modular Monolith
&lt;/h2>&lt;h3 id="-การออกแบบ-architecture">🏗️ การออกแบบ Architecture
&lt;/h3>&lt;p>&lt;strong>1. ใช้ Domain-Driven Design&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// แยก bounded context ชัดเจน&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">namespace&lt;/span> ECommerceApp.Modules.Orders.Domain
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">Order&lt;/span> { } &lt;span style="color:#75715e">// Order ใน Orders context&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">namespace&lt;/span> ECommerceApp.Modules.Catalog.Domain
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">Product&lt;/span> { } &lt;span style="color:#75715e">// Product ใน Catalog context&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Dependency Direction&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// ❌ ผิด: High-level module พึ่งพา low-level&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderService&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> SqlOrderRepository _repository; &lt;span style="color:#75715e">// Direct dependency&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// ✅ ถูก: Dependency Inversion&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderService&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">readonly&lt;/span> IOrderRepository _repository; &lt;span style="color:#75715e">// Interface dependency&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>3. Module Registration&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">ModuleExtensions&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> IServiceCollection AddOrdersModule(&lt;span style="color:#66d9ef">this&lt;/span> IServiceCollection services)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> services.AddScoped&amp;lt;IOrderRepository, SqlOrderRepository&amp;gt;();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> services.AddScoped&amp;lt;OrderService&amp;gt;();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> services.AddMediatR(&lt;span style="color:#66d9ef">typeof&lt;/span>(OrdersModule));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">return&lt;/span> services;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="-การจดการ-communication">🔄 การจัดการ Communication
&lt;/h3>&lt;p>&lt;strong>1. Synchronous Communication&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Interface-based communication สำหรับ immediate consistency&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">interface&lt;/span> &lt;span style="color:#a6e22e">IInventoryService&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Task&amp;lt;&lt;span style="color:#66d9ef">bool&lt;/span>&amp;gt; IsAvailableAsync(Guid productId, &lt;span style="color:#66d9ef">int&lt;/span> quantity);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Task ReserveAsync(Guid productId, &lt;span style="color:#66d9ef">int&lt;/span> quantity);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Asynchronous Communication&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Event-based communication สำหรับ eventual consistency&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderConfirmedEvent&lt;/span> : IDomainEvent
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> Guid OrderId { &lt;span style="color:#66d9ef">get&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> List&amp;lt;OrderItem&amp;gt; Items { &lt;span style="color:#66d9ef">get&lt;/span>; }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#a6e22e">
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#a6e22e">[EventHandler]&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">UpdateInventoryHandler&lt;/span> : IEventHandler&amp;lt;OrderConfirmedEvent&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task Handle(OrderConfirmedEvent @event)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Update inventory asynchronously&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="-การจดการ-data">🗃️ การจัดการ Data
&lt;/h3>&lt;p>&lt;strong>1. Shared Database with Schema Separation&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-sql" data-lang="sql">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">-- Orders schema
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&lt;/span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">SCHEMA&lt;/span> Orders;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">TABLE&lt;/span> Orders.Orders (...);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">TABLE&lt;/span> Orders.OrderItems (...);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">-- Products schema
&lt;/span>&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">&lt;/span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">SCHEMA&lt;/span> Products;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">TABLE&lt;/span> Products.Products (...);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">CREATE&lt;/span> &lt;span style="color:#66d9ef">TABLE&lt;/span> Products.Categories (...);
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Database per Module (Advanced)&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrdersDbContext&lt;/span> : DbContext
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">protected&lt;/span> &lt;span style="color:#66d9ef">override&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> OnConfiguring(DbContextOptionsBuilder optionsBuilder)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> optionsBuilder.UseSqlServer(connectionString, options =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.MigrationsHistoryTable(&lt;span style="color:#e6db74">&amp;#34;__OrdersMigrationsHistory&amp;#34;&lt;/span>, &lt;span style="color:#e6db74">&amp;#34;Orders&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> });
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="-การทดสอบ">🧪 การทดสอบ
&lt;/h3>&lt;p>&lt;strong>1. Module Integration Tests&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#a6e22e">[Test]&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task CreateOrder_WithValidProducts_ShouldSucceed()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Arrange&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> productService = ServiceProvider.GetService&amp;lt;IProductService&amp;gt;();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> orderService = ServiceProvider.GetService&amp;lt;IOrderService&amp;gt;();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Act&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> result = &lt;span style="color:#66d9ef">await&lt;/span> orderService.CreateOrderAsync(customerId, orderItems);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Assert&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Assert.IsTrue(result.IsSuccess);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Contract Tests&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#a6e22e">[Test]&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> ProductService_ShouldImplementIProductService()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#75715e">// Verify that ProductService implements all required interfaces&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">var&lt;/span> productService = &lt;span style="color:#66d9ef">new&lt;/span> ProductService();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> Assert.IsInstanceOf&amp;lt;IProductService&amp;gt;(productService);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h2 id="real-world-considerations">Real-world Considerations
&lt;/h2>&lt;h3 id="-production-enhancements">🔧 Production Enhancements
&lt;/h3>&lt;p>&lt;strong>1. Database Integration&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#75715e">// Replace in-memory repositories with Entity Framework Core&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>services.AddDbContext&amp;lt;OrdersDbContext&amp;gt;(options =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.UseSqlServer(connectionString));
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>services.AddScoped&amp;lt;IOrderRepository, EfOrderRepository&amp;gt;();
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Authentication &amp;amp; Authorization&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AddJwtBearer(options =&amp;gt; {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> options.TokenValidationParameters = &lt;span style="color:#66d9ef">new&lt;/span> TokenValidationParameters
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> ValidateIssuer = &lt;span style="color:#66d9ef">true&lt;/span>,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> ValidateAudience = &lt;span style="color:#66d9ef">true&lt;/span>,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> ValidateLifetime = &lt;span style="color:#66d9ef">true&lt;/span>,
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> ValidateIssuerSigningKey = &lt;span style="color:#66d9ef">true&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> };
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> });
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>3. Input Validation&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">CreateOrderCommandValidator&lt;/span> : AbstractValidator&amp;lt;CreateOrderCommand&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> CreateOrderCommandValidator()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RuleFor(x =&amp;gt; x.CustomerId).NotEmpty();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RuleFor(x =&amp;gt; x.Items).NotEmpty();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> RuleForEach(x =&amp;gt; x.Items).SetValidator(&lt;span style="color:#66d9ef">new&lt;/span> OrderItemValidator());
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>4. Error Handling&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">GlobalExceptionMiddleware&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">async&lt;/span> Task InvokeAsync(HttpContext context, RequestDelegate next)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">try&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> next(context);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">catch&lt;/span> (DomainException ex)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> HandleDomainExceptionAsync(context, ex);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">catch&lt;/span> (Exception ex)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> {
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">await&lt;/span> HandleGenericExceptionAsync(context, ex);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> }
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h3 id="-monitoring-และ-observability">📊 Monitoring และ Observability
&lt;/h3>&lt;p>&lt;strong>1. Health Checks&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>services.AddHealthChecks()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AddDbContextCheck&amp;lt;OrdersDbContext&amp;gt;(&lt;span style="color:#e6db74">&amp;#34;orders-db&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AddDbContextCheck&amp;lt;ProductsDbContext&amp;gt;(&lt;span style="color:#e6db74">&amp;#34;products-db&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .AddUrlGroup(&lt;span style="color:#66d9ef">new&lt;/span> Uri(&lt;span style="color:#e6db74">&amp;#34;https://external-api.com/health&amp;#34;&lt;/span>), &lt;span style="color:#e6db74">&amp;#34;external-api&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>2. Logging&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>services.AddSerilog((context, config) =&amp;gt;
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> config.ReadFrom.Configuration(context.Configuration)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .Enrich.WithProperty(&lt;span style="color:#e6db74">&amp;#34;Module&amp;#34;&lt;/span>, &lt;span style="color:#e6db74">&amp;#34;Orders&amp;#34;&lt;/span>)
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WriteTo.Console()
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .WriteTo.File(&lt;span style="color:#e6db74">&amp;#34;logs/orders-.log&amp;#34;&lt;/span>, rollingInterval: RollingInterval.Day);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>});
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>&lt;strong>3. Metrics&lt;/strong>&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;">&lt;code class="language-csharp" data-lang="csharp">&lt;span style="display:flex;">&lt;span>&lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">class&lt;/span> &lt;span style="color:#a6e22e">OrderMetrics&lt;/span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>{
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">private&lt;/span> &lt;span style="color:#66d9ef">static&lt;/span> &lt;span style="color:#66d9ef">readonly&lt;/span> Counter OrdersCreated = Metrics
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> .CreateCounter(&lt;span style="color:#e6db74">&amp;#34;orders_created_total&amp;#34;&lt;/span>, &lt;span style="color:#e6db74">&amp;#34;Total number of orders created&amp;#34;&lt;/span>);
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span> &lt;span style="color:#66d9ef">public&lt;/span> &lt;span style="color:#66d9ef">void&lt;/span> IncrementOrdersCreated() =&amp;gt; OrdersCreated.Inc();
&lt;/span>&lt;/span>&lt;span style="display:flex;">&lt;span>}
&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;h2 id="ขอสรป">ข้อสรุป
&lt;/h2>&lt;p>Modular Monolith เป็นสถาปัตยกรรมที่เหมาะสมสำหรับองค์กรหลายประเภท โดยเฉพาะทีมที่ต้องการประโยชน์ของ modularity แต่ยังไม่พร้อมสำหรับความซับซ้อนของ microservices&lt;/p>
&lt;h3 id="-key-takeaways">🔑 Key Takeaways:
&lt;/h3>&lt;p>&lt;strong>1. เริ่มต้นอย่างเรียบง่าย&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ใช้ Modular Monolith เป็นจุดเริ่มต้น&lt;/li>
&lt;li>Focus บน domain modeling และ clean boundaries&lt;/li>
&lt;li>ปรับปรุงแบบค่อยเป็นค่อยไป&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>2. ออกแบบให้พร้อมสำหรับการเปลี่ยนแปลง&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ใช้ interfaces สำหรับ inter-module communication&lt;/li>
&lt;li>Implement domain events infrastructure&lt;/li>
&lt;li>เตรียม monitoring และ observability&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>3. วางแผน Migration Path&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>ระบุ modules ที่อาจต้อง extract ในอนาคต&lt;/li>
&lt;li>ออกแบบ data access patterns ที่รองรับการแยก&lt;/li>
&lt;li>สร้าง culture ของ modular thinking&lt;/li>
&lt;/ul>
&lt;p>&lt;strong>4. Focus บน Business Value&lt;/strong>&lt;/p>
&lt;ul>
&lt;li>อย่าให้ technical complexity บดบัง business goals&lt;/li>
&lt;li>ใช้สถาปัตยกรรมที่เหมาะสมกับทีมและ context&lt;/li>
&lt;li>Remember: Architecture คือ means ไม่ใช่ end&lt;/li>
&lt;/ul>
&lt;h3 id="-next-steps">🚀 Next Steps:
&lt;/h3>&lt;ol>
&lt;li>&lt;strong>ศึกษาโค้ด&lt;/strong>: ดู &lt;a class="link" href="https://github.com/suriyasonp/modular-monolith-ecommerce" target="_blank" rel="noopener"
>repository&lt;/a> เพื่อเข้าใจ implementation details&lt;/li>
&lt;li>&lt;strong>ทดลองเรียกใช้&lt;/strong>: รัน &lt;code>./test-api.sh&lt;/code> เพื่อดูฟีเจอร์ทั้งหมด&lt;/li>
&lt;li>&lt;strong>เพิ่มฟีเจอร์&lt;/strong>: ลองสร้างโมดูลใหม่ตามแพทเทิร์นที่กำหนด&lt;/li>
&lt;li>&lt;strong>นำไปใช้จริง&lt;/strong>: ปรับแต่งสำหรับ production environment&lt;/li>
&lt;/ol>
&lt;p>Modular Monolith ไม่ใช่ silver bullet แต่เป็นเครื่องมือที่มีประสิทธิภาพสำหรับการสร้างระบบที่ maintainable, scalable และ evolvable ในระยะยาว&lt;/p>
&lt;h2 id="แหลงอางอง">แหล่งอ้างอิง
&lt;/h2>&lt;ol>
&lt;li>&lt;a class="link" href="https://github.com/suriyasonp/modular-monolith-ecommerce" target="_blank" rel="noopener"
>Modular Monolith E-commerce Repository&lt;/a> - ตัวอย่างการ implement&lt;/li>
&lt;li>&lt;a class="link" href="https://domainlanguage.com/ddd/" target="_blank" rel="noopener"
>Domain-Driven Design&lt;/a> - Eric Evans&lt;/li>
&lt;li>&lt;a class="link" href="https://www.kamilgrzybek.com/design/modular-monolith-primer/" target="_blank" rel="noopener"
>Modular Monoliths&lt;/a> - Kamil Grzybek&lt;/li>
&lt;li>&lt;a class="link" href="https://samnewman.io/books/building_microservices/" target="_blank" rel="noopener"
>Building Microservices&lt;/a> - Sam Newman&lt;/li>
&lt;li>&lt;a class="link" href="https://blog.cleancoder.com/uncle-bob/2012/08/13/the-clean-architecture.html" target="_blank" rel="noopener"
>Clean Architecture&lt;/a> - Robert C. Martin&lt;/li>
&lt;li>&lt;a class="link" href="https://docs.microsoft.com/en-us/aspnet/core/" target="_blank" rel="noopener"
>ASP.NET Core Documentation&lt;/a> - Microsoft Docs&lt;/li>
&lt;li>&lt;a class="link" href="https://blog.bytebytego.com/p/monolith-vs-microservices-vs-modular" target="_blank" rel="noopener"
>Monolith vs Microservices vs Modular Monoliths&lt;/a> - Monolith vs Microservices vs Modular Monoliths&lt;/li>
&lt;/ol>
&lt;hr></description></item></channel></rss>