Author’s Note: After announcing the Modern Web App (MWA) pattern (https://devblogs.microsoft.com/java/introducing-the-modern-web-app-mwa-pattern-for-java), we now focus on one of its many pattens – the Strangler Fig pattern. This pattern incrementally modernizes a legacy Java application by “strangling” targeted areas and replacing them with new microservices or minimal web apps. We’ll explore an example showing how to integrate this approach and smoothly transition from a monolith to modern microservices.
MWA pattern: Strangler Fig
Introduction
In today’s fast-paced technological landscape, modernizing Java applications is crucial for achieving scalability, maintainability, and performance. Legacy systems often struggle to keep up with the demands of modern users and evolving business requirements. By adopting modern architectural patterns, organizations can ensure their applications remain competitive and capable of handling increased workloads.
The Strangler Fig Pattern: A Strategy for Incremental Modernization
As applications age, the tools, technologies, and architectures they rely on can become outdated. Introducing new features can increase complexity, making these systems harder to maintain and evolve. Completely rewriting a complex system is often a massive and risky undertaking. The Strangler Fig Pattern offers a solution by allowing you to incrementally migrate a legacy system. This is achieved by gradually replacing specific pieces of functionality with new applications and services.
Inspired by the strangler fig tree that grows around and eventually replaces its host, this pattern involves creating a façade that intercepts requests to the backend legacy system. The façade routes these requests either to the legacy application or the new services. This approach enables existing features to be migrated gradually, while consumers continue using the same interface, unaware of the changes happening behind the scenes.
By gradually replacing the legacy system’s features, the new system eventually encompasses all functionalities, allowing you to safely decommission the old system.
Read Martin Fowler’s original article on the Strangler Fig Application here.
Example
Next let’s look at a typical use case where the Strangler Fig pattern would make a good fit. Imagine you’re migrating a /customer
endpoint from a legacy system to a modernized architecture. You can deploy a router that directs requests to either the old or new implementation based on specific conditions.
Java Code Example
@RestController
@RequestMapping("/api")
public class CustomerRouterController {
private final LegacyCustomerService legacyService;
private final ModernCustomerService modernService;
public CustomerRouterController(LegacyCustomerService legacyService, ModernCustomerService modernService) {
this.legacyService = legacyService;
this.modernService = modernService;
}
@GetMapping("/customer")
public ResponseEntity<Customer> getCustomer(@RequestParam String id) {
if (isModernCustomer(id)) {
Customer customer = modernService.getCustomerById(id);
return ResponseEntity.ok(customer);
}
Customer customer = legacyService.getCustomerById(id);
return ResponseEntity.ok(customer);
}
private boolean isModernCustomer(String id) {
// Logic to determine if the request should be routed to the modern service
return id.startsWith("MODERN_");
}
}
This example uses a centralized router to direct requests based on specific criteria, such as the format of the customer ID. It enables both legacy and modern systems to coexist during the transition period. Follow these steps to run the Spring Boot example demonstrating the Strangler Fig pattern.
Prerequisites
Ensure you have the following tools installed:
- JDK 17: Download
- Apache Maven: Download
- Visual Studio Code: Download
- Java Extension Pack for Visual Studio Code: Marketplace
Steps to Run
-
Clone the Repository
git clone https://github.com/roryp/stranglerfig.git cd stranglerfig
-
Build the Project
mvn clean install
-
Run the Application
mvn spring-boot:run
The application starts on port 8080 by default.
-
Test the Application
Send requests to the
/api/customer
endpoint:-
Modern Customer ID
curl http://localhost:8080/api/customer?id=MODERN_1
This request is routed to the modern customer service based on the customer ID prefix e.g “MODERN_”.
-
Legacy Customer IDs
curl http://localhost:8080/api/customer?id=LEGACY_1
These requests are routed to the legacy customer service based on the customer ID prefix e.g “LEGACY_”.
-
Modern Java Web App (MWA) Pattern
Now that the Strangler Fig pattern has been introduced, it can be combined with other techniques to form the MWA Pattern. The MWA pattern emphasizes resilience, security, and scalability, leveraging cloud-native services and best practices. To centrally manage application settings, it uses Azure App Configuration, which supports feature toggling and the gradual rollout of new features.
Next, let’s see how to add Azure App Configuration to a typical Spring Boot application:
-
Add the Dependency to your
pom.xml
:<dependency> <groupId>com.azure.spring</groupId> <artifactId>azure-spring-cloud-feature-management-web</artifactId> <version>2.3.0</version> </dependency>
-
Configure Connection Settings in
application.properties
:spring.cloud.azure.appconfiguration.stores[0].endpoint=https://<your-app-config-name>.azconfig.io spring.cloud.azure.appconfiguration.stores[0].connection-string=<your-connection-string>
-
Example Feature Flag
@RestController @RequestMapping("/api") public class FeatureFlagController { @GetMapping("/feature") @FeatureGate(feature = "BetaFeature") public ResponseEntity<String> getFeature() { return ResponseEntity.ok("Beta Feature is enabled!"); } }
With this example, the /feature
endpoint will only be active if the BetaFeature
flag is enabled in Azure App Configuration.
Next, explore our end-to-end (with great focus on Strangler Fig and Azure App Config) MWA reference example:
- Clone the MWA respository
- Deploy the App using
azd up
- Real-time migrate an Azure App using the Strangler Fig Pattern – demo.md
Conclusion
Modernizing Java applications using the Strangler Fig pattern and the Modern Web App (MWA) pattern provides a strategic approach to incrementally evolve legacy systems. By leveraging these patterns, organizations can achieve scalability, maintainability, and performance while minimizing risks and ensuring continuous availability. Start your modernization journey today by adopting these proven architectural patterns with the help of the Modern Java Web App Patterns Repository
Useful Links
- Deploy the Reference App: Experience all MWA principles in action by deploying the full production-grade Java application to Azure. Visit the MWA GitHub repository for more information.
- Explore In-Depth Documentation: Learn more about MWA through comprehensive documentation on Microsoft Learn.
- Try Out MWA Patterns: Discover and experiment with the MWA patterns to modernize and scale your Java applications
- Source code for this blog: roryp/stranglerfig: Modernizing Java Applications with the Strangler and Modern Web App Patterns
0 comments
Be the first to start the discussion.