Creating a Simple CRUD API with Spring Boot and In-Memory Database

Below is a step-by-step guide to create a simple CRUD API using Spring Boot and an in-memory database like H2. We will use Maven to manage the project’s dependencies.

1. Set Up Your Maven Project

Create a new Maven project or directory for the Spring Boot application.

pom.xml

Here is the file with the necessary dependencies: pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <groupId>com.example</groupId>
    <artifactId>crud-api</artifactId>
    <version>1.0-SNAPSHOT</version>
    <packaging>jar</packaging>
    <name>Spring Boot CRUD API</name>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>3.5.3</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <java.version>21</java.version>
    </properties>

    <dependencies>
        <!-- Spring Boot Starter Web -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!-- Spring Boot Starter JPA -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>

        <!-- H2 Database (In-memory database) -->
        <dependency>
            <groupId>com.h2database</groupId>
            <artifactId>h2</artifactId>
            <scope>runtime</scope>
        </dependency>

        <!-- Lombok for simplicity -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <scope>provided</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <!-- Maven Spring Boot plugin -->
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>
</project>

2. Create the Application Entry Point

src/main/java/org/kodejava/crudapi/CrudApiApplication.java

Create the main class for your Spring Boot application.

package org.kodejava.crudapi;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class CrudApiApplication {

    public static void main(String[] args) {
        SpringApplication.run(CrudApiApplication.class, args);
    }
}

3. Define the Entity

Create a User entity to model your data.

src/main/java/org/kodejava/crudapi/model/User.java

package org.kodejava.crudapi.model;

import jakarta.persistence.*;

import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@NoArgsConstructor
@Entity
@Table(name = "users")
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    private String name;

    private String email;
}

4. Create the Repository

The repository will handle database operations.

src/main/java/org/kodejava/crudapi/repository/UserRepository.java

package org.kodejava.crudapi.repository;

import com.example.crudapi.model.User;
import org.springframework.data.jpa.repository.JpaRepository;

public interface UserRepository extends JpaRepository<User, Long> {
}

5. Define the Service

The service layer will handle business logic.

src/main/java/org/kodejava/crudapi/service/UserService.java

package org.kodejava.crudapi.service;

import com.example.crudapi.model.User;
import com.example.crudapi.repository.UserRepository;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class UserService {

    private final UserRepository userRepository;

    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    public List<User> getAllUsers() {
        return userRepository.findAll();
    }

    public User getUserById(Long id) {
        return userRepository.findById(id).orElseThrow(() -> new RuntimeException("User not found"));
    }

    public User createUser(User user) {
        return userRepository.save(user);
    }

    public User updateUser(Long id, User user) {
        User existingUser = getUserById(id);
        existingUser.setName(user.getName());
        existingUser.setEmail(user.getEmail());
        return userRepository.save(existingUser);
    }

    public void deleteUser(Long id) {
        userRepository.deleteById(id);
    }
}

6. Create the Controller

The controller defines API endpoints for CRUD operations.

src/main/java/org/kodejava/crudapi/controller/UserController.java

package org.kodejava.crudapi.controller;

import com.example.crudapi.model.User;
import com.example.crudapi.service.UserService;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;

import java.util.List;

@RestController
@RequestMapping("/api/users")
public class UserController {

    private final UserService userService;

    public UserController(UserService userService) {
        this.userService = userService;
    }

    @GetMapping
    public ResponseEntity<List<User>> getAllUsers() {
        return ResponseEntity.ok(userService.getAllUsers());
    }

    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable Long id) {
        return ResponseEntity.ok(userService.getUserById(id));
    }

    @PostMapping
    public ResponseEntity<User> createUser(@RequestBody User user) {
        return ResponseEntity.status(HttpStatus.CREATED).body(userService.createUser(user));
    }

    @PutMapping("/{id}")
    public ResponseEntity<User> updateUser(@PathVariable Long id, @RequestBody User user) {
        return ResponseEntity.ok(userService.updateUser(id, user));
    }

    @DeleteMapping("/{id}")
    public ResponseEntity<Void> deleteUser(@PathVariable Long id) {
        userService.deleteUser(id);
        return ResponseEntity.noContent().build();
    }
}

7. Configure Application Properties

Configure the H2 database in . application.properties

src/main/resources/application.properties

# H2 In-Memory Database Settings
spring.datasource.url=jdbc:h2:mem:testdb
spring.datasource.driverClassName=org.h2.Driver
spring.datasource.username=sa
spring.datasource.password=password
spring.jpa.database-platform=org.hibernate.dialect.H2Dialect
spring.h2.console.enabled=true
spring.h2.console.path=/h2-console

8. Test the Application

Run your Spring Boot application and test the REST APIs using tools like Postman or curl.

  • To get all users:
  GET http://localhost:8080/api/users
  • To get a user by ID:
  GET http://localhost:8080/api/users/{id}
  • To create a user:
  POST http://localhost:8080/api/users
  Content-Type: application/json
  {
      "name": "John Doe",
      "email": "[email protected]"
  }
  • To update a user:
  PUT http://localhost:8080/api/users/{id}
  Content-Type: application/json
  {
      "name": "Jane Doe",
      "email": "[email protected]"
  }
  • To delete a user:
  DELETE http://localhost:8080/api/users/{id}

9. Access the H2 Console

You can access the H2 database console at http://localhost:8080/h2-console.

  • JDBC URL: jdbc:h2:mem:testdb
  • Username: sa
  • Password: password

That’s it! You now have a fully functional CRUD API with Spring Boot and an in-memory H2 database.

How do I write and register a custom BeanFactoryPostProcessor?

A BeanFactoryPostProcessor is a special type of bean in Spring that allows you to modify the application context’s internal bean definitions before they are instantiated. To write and register a custom BeanFactoryPostProcessor, you can follow these steps:

Step 1: Implement the BeanFactoryPostProcessor Interface

You need to create a class that implements the BeanFactoryPostProcessor interface. This interface requires you to implement the postProcessBeanFactory() method, where you can manipulate bean definitions.
Here’s an example of a custom BeanFactoryPostProcessor implementation:

import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.stereotype.Component;

@Component
public class CustomBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        // Retrieve a specific bean definition
        BeanDefinition beanDefinition = beanFactory.getBeanDefinition("myBean");

        // Modify the bean definition as needed (e.g., change the bean's scope)
        beanDefinition.setScope(BeanDefinition.SCOPE_PROTOTYPE);

        // You can also log or manipulate other metadata, such as property values
        System.out.println("CustomBeanFactoryPostProcessor is modifying the bean definition for 'myBean'");
    }
}

Step 2: Register the Custom BeanFactoryPostProcessor

There are two main ways to register your BeanFactoryPostProcessor:

a. Use @Component Annotation

If you annotate your custom post-processor class with @Component, Spring automatically detects and registers it during classpath scanning (if component scanning is enabled).

@Component
public class CustomBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
    // Implementation remains the same
}

Ensure that your application context is set up to scan the package containing this class.

b. Register Explicitly in a Configuration Class

You can define the BeanFactoryPostProcessor explicitly in a Spring @Configuration class:

import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyConfig {

    @Bean
    public BeanFactoryPostProcessor customBeanFactoryPostProcessor() {
        return new CustomBeanFactoryPostProcessor();
    }
}

Step 3: Test the Custom BeanFactoryPostProcessor

To test if your BeanFactoryPostProcessor is working, ensure that you have a bean with the name or properties you intend to modify. For example:

import org.springframework.stereotype.Component;

@Component("myBean")
public class MyBean {
    public MyBean() {
        System.out.println("MyBean instance created");
    }
}

Run your application, and you’ll see the custom modifications applied before MyBean is created. The prototype scope will now be in effect if that’s the modification you implemented.

Notes:

  1. Execution Order: If there are multiple BeanFactoryPostProcessor instances, Spring executes them in the order determined by their priority (@Order annotation or PriorityOrdered interface).
  2. Responsibility: BeanFactoryPostProcessor only processes BeanDefinition objects; it doesn’t create or initialize beans.
  3. Difference from BeanPostProcessor: Unlike a BeanPostProcessor, which works on bean instances after they are created, a BeanFactoryPostProcessor works with bean definitions before beans are instantiated.

This concludes the steps to write, register, and use a custom BeanFactoryPostProcessor.

Spring Boot Configuration Essentials: application.properties vs application.yml

In Spring Boot, both application.properties and application.yml are configuration files used to define application settings, but they differ mainly in syntax and structure. Here’s a detailed comparison to help you understand their essentials:

1. File Format

application.properties:

  • A key-value pair format, where each property is defined on a new line.
  • A simple and widely used format for configuration files.

Example:

server.port=8080
spring.datasource.url=jdbc:mysql://localhost:3306/mydb
spring.datasource.username=admin
spring.datasource.password=secret

application.yml:

  • A hierarchical data format popular for its readability, using indentation to denote levels.
  • Based on YAML syntax, it is more concise for complex hierarchical configurations.

Example:

server:
  port: 8080

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/mydb
    username: admin
    password: secret

2. Structure and Readability

  • application.properties: Flat structure; can get lengthy and harder to read with nested configurations.
  • application.yml: More concise and easier to maintain hierarchical data (e.g., grouping related properties).

For instance:

  • Nested configuration in application.properties:
  spring.datasource.url=jdbc:mysql://localhost:3306/mydb
  spring.datasource.username=admin
  spring.datasource.password=secret
  spring.jpa.hibernate.ddl-auto=update
  • Nested configuration in application.yml:
  spring:
    datasource:
      url: jdbc:mysql://localhost:3306/mydb
      username: admin
      password: secret
    jpa:
      hibernate:
        ddl-auto: update

3. Multi-Profile Support

  • Both formats support profiles (e.g., application-dev.properties or application-dev.yml).
  • However, in application.yml, profiles are defined more elegantly using spring.profiles.

Example in application.properties:

# application-dev.properties
spring.datasource.url=jdbc:mysql://localhost:3306/devdb

Example in application.yml:

spring:
  profiles:
    active: dev

---
spring:
  profiles: dev
  datasource:
    url: jdbc:mysql://localhost:3306/devdb

The --- in YAML separates different profiles, while spring.profiles specifies the active one.

4. Comments

  • application.properties: Use # for single-line comments.
    Example:
  # This sets the server's port
  server.port=8080
  • application.yml: Also uses # for comments.
    Example:
  # This sets the server's port
  server:
    port: 8080

5. Tools and Validation

  • YAML syntax errors (like incorrect indentation) are harder to debug compared to properties.
  • Most IDEs (like IntelliJ IDEA) provide excellent support for both formats, with syntax highlighting and validation tools.

6. When to Use Which?

  • Use application.properties:
    • If you prefer simplicity and are comfortable with key-value pairs.
    • For flat and straightforward configurations.
  • Use application.yml:
    • If you want a more structured and readable format.
    • For more complex hierarchical configurations or when working with deeply nested values.

7. Mixing Both

  • Spring Boot supports both application.properties and application.yml in the same project, but it’s recommended to stick to one for consistency.
  • If both exist, application.properties takes precedence over application.yml (as per Spring Boot’s default property source order).

Summary Table:

Aspect application.properties application.yml
Format Key-value pair Hierarchical YAML format
Readability Harder for nested values Easy to read and maintain with indentation
Profiles Separate files for each profile Inline profiles with --- separator
Error handling Simple; less prone to errors Easy to misconfigure due to indentation
Preference Flat structure Complex or hierarchical configuration

How do I implement custom bean post-processors in Spring Framework?

In the Spring Framework, a custom BeanPostProcessor can be implemented to apply custom logic before and after initialization of each bean in the application context. This is primarily useful for scenarios where you need to modify bean instances, validate configurations, or apply custom processing during the initialization phase of a bean’s lifecycle.

Here’s how you can implement and use a custom BeanPostProcessor:


Steps to Implement a Custom BeanPostProcessor:

  1. Create a Class that Implements BeanPostProcessor Interface:
    The BeanPostProcessor interface provides two methods:

    • postProcessBeforeInitialization(Object bean, String beanName)
    • postProcessAfterInitialization(Object bean, String beanName)
  2. Override the Methods:
    In most cases, you’ll override one or both of these methods to define your custom logic:

    • postProcessBeforeInitialization: Executed before the bean’s @PostConstruct method (if any) and the initialization.
    • postProcessAfterInitialization: Executed after the initialization phase.
  3. Register the BeanPostProcessor in the Application Context:
    The custom BeanPostProcessor needs to be registered as a Spring bean so it can intercept the bean lifecycle.

Example Implementation

Here’s an example of creating a custom BeanPostProcessor:

import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component;

@Component
public class MyCustomBeanPostProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) {
        System.out.println("Before Initialization of bean: " + beanName);
        // You can modify the bean instance here if required
        return bean; // Return the same or a wrapped bean instance
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) {
        System.out.println("After Initialization of bean: " + beanName);
        // You can modify the bean instance here if required
        return bean; // Return the same or a wrapped bean instance
    }
}

Key Points:

  1. Registering the Processor:
    • The @Component annotation registers the MyCustomBeanPostProcessor bean in the Spring context automatically.
    • Alternatively, you can declare it explicitly in a @Configuration class using @Bean.
  2. Scope of Processing:
    • A BeanPostProcessor is applied to all beans in the application context.
    • You can put conditions within the if block (e.g., check beanName or bean.getClass() type) to restrict processing to specific beans.
  3. Returning the Bean:
    • Always return the bean instance (or a proxy/wrapped version) from the methods.
    • Altering or replacing the bean instance might affect its behavior in subsequent phases.
  4. Execution Order:
    • If there are multiple BeanPostProcessor implementations, you can set execution order by implementing the Ordered interface or using the @Order annotation.

Example with Conditional Processing:

If you want your BeanPostProcessor to act only on specific bean types or names (e.g., beans of a specific class):

import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component;

@Component
public class ExampleConditionalBeanPostProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) {
        if (bean instanceof MySpecialBean) {
            System.out.println("Custom logic for MySpecialBean before initialization: " + beanName);
        }
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) {
        if (bean instanceof MySpecialBean) {
            System.out.println("Custom logic for MySpecialBean after initialization: " + beanName);
        }
        return bean;
    }
}

Use Cases for BeanPostProcessor:

  • Modify or wrap bean instances dynamically (e.g., for proxy generation).
  • Log initialization phases of beans.
  • Inject additional dependencies into specific beans post-creation.
  • Validate or check bean configurations.

The BeanPostProcessor is a powerful way to hook into the Spring container’s lifecycle processing and apply cross-cutting concerns without modifying individual bean definitions directly.

Using Spring Boot DevTools for Faster Development and Live Reload

Spring Boot DevTools is a valuable tool for speeding up the development process by enabling features like automatic application restart and live reload of web content. It is specifically designed to improve the development experience by reducing the time required to restart the application during testing and debugging.

Here is a quick guide to using Spring Boot DevTools for faster development and live reload:


1. Add DevTools Dependency

To use Spring Boot DevTools, include it in your pom.xml (Maven) or build.gradle (Gradle) file.

Using Maven:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-devtools</artifactId>
    <optional>true</optional>
</dependency>

Using Gradle:

implementation 'org.springframework.boot:spring-boot-devtools'

2. Automatic Restart

Spring Boot DevTools triggers an automatic application restart whenever files in the classpath are modified. It uses two classloaders—one for the static resources and one for the application classes—enabling a fast reload experience.

  • Restart Triggering: Files located under /src/main/resources/, /src/main/java/, or any other classpath resources automatically restart the application upon modification.
  • Excluding Certain Files from Restart: You can exclude specific file patterns using the property:
spring.devtools.restart.exclude=static/**,public/**

3. Live Reload (Optional with Browser)

Spring Boot DevTools integrates with LiveReload, so front-end changes (e.g., HTML, CSS, or JavaScript) trigger an automatic browser reload.

Steps for LiveReload:

  1. Install a LiveReload extension in your browser (available for Chrome, Firefox, etc.).
  2. DevTools will automatically enable LiveReload if the extension is active.

If you want to disable the LiveReload capability, use the following property:

spring.devtools.livereload.enabled=false

4. Property Defaults in Development vs. Production

DevTools provides sensible defaults for development environments that are different from production. For instance:

  • Caching is disabled for templates (e.g., Thymeleaf, FreeMarker, etc.).
  • Hibernate auto-detection for changes in the database schema is enabled.

If you want to customize DevTools properties, you can use a dedicated application-dev.properties profile.

5. How to Enable Conditional DevTools Behavior

To avoid shipping DevTools to production, mark it as optional=true in Maven or use a developmentOnly configuration in Gradle.

Example for Gradle:

developmentOnly 'org.springframework.boot:spring-boot-devtools'

6. Disable Restart in Specific Scenarios

If you don’t want restart functionality during your development, you can disable it with the property:

spring.devtools.restart.enabled=false

7. Trigger a Manual Restart

If you want to trigger a restart manually during development, you can:

  • Add or remove files in the /META-INF/spring-devtools.properties directory to trigger a restart.

8. Example Use Case: Thymeleaf or Front-End Modification

If you’re using Thymeleaf templates in a Spring Boot application for the web frontend:

  • Modify an HTML file under /src/main/resources/templates.
  • The browser will refresh automatically (if LiveReload is enabled). You’ll instantly see your changes without manually restarting or refreshing.

Important Notes:

  • Spring Boot DevTools is solely for development purposes and should not be packaged into your production build.
  • If you’re running in Docker or a cloud environment, ensure that file watchers are properly configured for file changes.

Incorporating Spring Boot DevTools provides a more efficient development experience by automating repetitive actions and making live coding efforts more seamless!