Saturday, March 8, 2025
Best Java Libraries for XML Data Processing
Saturday, March 1, 2025
Upgrading to Java 21 and Spring Boot 3: A Comprehensive Guide
The transition from Java 17 to Java 21, paired with an upgrade to Spring Boot 3, is a transformative step for modern application development. This blog post shares the detailed insights, challenges, and solutions from upgrading various services and Lambda functions to these latest versions. We’ll explore the evolution of dependencies, dive into troubleshooting steps, discuss resolutions for common issues, and provide actionable takeaways to help you succeed in your own upgrade journey.
Overview of the Upgrade
The upgrade process entailed moving applications from Java 17 to Java 21 and aligning them with Spring Boot 3. This wasn’t a simple plug-and-play operation—it required careful updates to dependencies, adjustments to configurations, and tweaks to codebases to ensure everything worked harmoniously with the new versions. The process touched multiple components, including core services and AWS Lambda functions, each requiring its own set of changes.
Dependency Evolution
The upgrade unfolded in stages, with dependencies evolving iteratively as issues surfaced and were resolved. Here’s how the key components changed over time:
Java: Initially running on version 17, the leap to Java 21 introduced a "Major version 65" issue, signaling bytecode incompatibility with older tools. This necessitated updates to other dependencies to align with Java 21’s requirements.
Gradle: Starting at version 7.2, we upgraded to 8.1 early in the process to leverage its improved features and compatibility with Java 21. However, this shift triggered initial compile errors, which we addressed as part of broader dependency updates.
Spring Boot: We began with version 2.7.3 but quickly encountered limitations. An intermediate step to 2.7.18 resolved some issues, but JUnit and acceptance test failures persisted. The final move to Spring Boot 3.3.3 was essential to fully support Java 21 and handle the significant shift from javax to jakarta namespaces.
AWS SDK v1: Version 1.12.139 was in use initially, but it proved incompatible with Java 21. We phased it out entirely, relying instead on AWS SDK v2.
AWS SDK v2: Starting at 2.17.162, this remained stable throughout the upgrade, though we later validated its compatibility with the final configuration.
LocalStack: We started with version 1.17.1 for local testing. As issues emerged with Spring Boot 3, we upgraded to 1.20.1 and eventually aligned the LocalStack Docker image to version 3.0.0 for better test reliability.
LocalStack Docker: The initial version, 0.14.0, was outdated for our needs. Upgrading to 3.0.0 ensured compatibility with the updated LocalStack and Spring Boot 3.
Lombok Plugin: Version 6.4.3 caused compile errors with Java 21. Upgrading to 8.10 resolved these issues and ensured smooth integration with the new Java version.
AWS Spring: We began with version 2.4.4, which worked with Spring Boot 2.x. The move to Spring Boot 3 required an update to version 3.1.1 to maintain AWS integration.
Spring Cloud AWS Messaging: Also at 2.4.4 initially, this dependency was ultimately removed as we streamlined our AWS interactions with SDK v2.
Each step in this evolution addressed specific pain points—whether it was compilation failures, test issues, or runtime errors—bringing us closer to a fully functional Java 21 and Spring Boot 3 setup.
Throughout the upgrade, several issues emerged that required targeted solutions. Here’s a detailed look at what we encountered and how we resolved them:
Solution: We updated the configuration key from management.metrics.export.statsd.enabled to management.statsd.metrics.export.enabled and explicitly enabled it in the application’s YAML file:
management:
statsd:
metrics:
export:
enabled: true
This adjustment silenced the log spam and restored proper metrics behavior.
Conclusion
Upgrading to Java 21 and Spring Boot 3 is a complex but rewarding endeavor. By navigating challenges like log spam from PortUnreachableException, Gradle task misconfigurations, and AWS SDK transitions, you can modernize your applications for improved performance and maintainability. This guide offers a detailed roadmap to help you avoid common pitfalls and achieve a successful upgrade.
Tuesday, March 19, 2024
Simplifying Docker Deployment with PM2
As you know, PM2 is a daemon process manager that allows you to keep applications online. Many times, you may be running your service inside a Docker image. The service can be written in any language, such as Node.js, Java, etc. Below is the shell script that can be used to deploy your service. It can be added and used in your CI/CD pipeline.
#!/usr/bin/env bash | |
set -euo pipefail | |
# Variables | |
IMAGE_NAME="my_image_name" | |
CONTAINER_NAME="my_container_name" | |
TAG="latest" | |
# Build Docker image | |
docker build -t "${IMAGE_NAME}" . | |
# Check if the container is already running | |
if [[ $(pm2 pid "$CONTAINER_NAME") ]]; then | |
# Container exists, update and restart | |
echo "Container ${CONTAINER_NAME} already exists. Updating and restarting..." | |
# Stop the existing container | |
pm2 stop ${CONTAINER_NAME} | |
# Remove the existing container | |
docker rm -f ${CONTAINER_NAME} | |
# Remove the process | |
pm2 delete ${CONTAINER_NAME} | |
# Save process after delete | |
pm2 save | |
else | |
# Container doesn't exist, start a new one | |
echo "Container ${CONTAINER_NAME} doesn't exist. Starting a new one..." | |
fi | |
# Run the Docker container | |
pm2 start docker --name ${CONTAINER_NAME} -- run --name ${CONTAINER_NAME} -p 3000:3000 -e ENV="${ENV}" --env-file "/root/backend.env_${ENV}" --restart=always ${IMAGE_NAME}:${TAG} | |
# Save process in pm2 | |
pm2 save | |
# Remove dangling images | |
docker system prune -f |
Sunday, December 31, 2023
AWS - Get quicksight embed url using JavaScript SDK V3
In the realm of data-driven solutions, AWS QuickSight offers a robust platform for crafting dynamic and insightful dashboards. Embedding these dashboards directly into your applications adds a layer of accessibility and convenience. This guide walks you through the process of obtaining a secure QuickSight embed URL using JavaScript SDK V3, suitable for both Node.js backend and Lambda functions. Before proceeding, ensure your QuickSight dashboard is created and shared with the intended audience.
Prerequisites:
Make sure you've completed the following preliminary steps:
Dashboard Setup:Friday, June 2, 2023
Docker - err: exec: "docker-credential-desktop": executable file not found in $PATH, out: ``
Introduction:
Docker has revolutionized the way we package, distribute, and deploy applications. However, like any other technology, it is not immune to errors and issues. One such error that I encountered recently while working with Docker on my Mac was the "error getting credentials - executable file not found" error. In this blog post, I will share my experience with this error and provide a solution that worked for me.
The Error:
When attempting to build a Docker image or perform a Docker login, I encountered the following error message:
openjdk:17: error getting credentials - err: exec: "docker-credential-desktop": executable file not found in $PATH, out: ``
Error saving credentials: error storing credentials - err: exec: "docker-credential-desktop": executable file not found in $PATH, out: ``
Diagnosis:
The error message suggests that the Docker daemon was unable to locate the "docker-credential-desktop" executable file in the $PATH environment variable. This executable is responsible for retrieving Docker credentials and authentication.