Skip to content

Latest commit

 

History

History
153 lines (131 loc) · 5.44 KB

File metadata and controls

153 lines (131 loc) · 5.44 KB

MapStruct Mapping Facade Quality Gate Status

Overview

This is a generic Facade over MapStruct mapping Library. MapStruct gives as Java POJOs compile-time mapping, we give you a single entry and use point to all registered MapStruct mappers in Spring Boot environment. We use some spring-based reflection here to collect mapping metadata and register mappings.

How it works?

Let's say you have 2 Java POJOs, Car and CarDto.

public class Car {
    private final String name;
    private final int year;

    public Car(String name, int year) { // implementation }
}

public class CarDto {
    private final String name;
    private final int year;

    public CarDto(String name, int year) { // implementation }
}

You have an instance of Car and need to convert it into CarDto. In the first place, let's declare a mapper in MapStruct style and extend it from UnidirectionalMapper interface. It will give us unidirectional mapping abstractions. Don't forget to annotate it with MapStruct's @Mapper annotation and set componentModel to "spring".

@Mapper(componentModel = "spring")
public interface CarMapper extends UnidirectionalMapper<Car, CarDto> {}

That's it. We've configured and registered a mapping! Now we can inject MappingFacade and map our POJOs.

@Component
public class OurBusinessBean {
    
    @Autowird
    private MappingFacade mappingFacade;
    
    public CarDto doMap(Car car) {
        return mappingFacade.map(car, CarDto.class);
    }
}

We also support polymorphic type mappings, e.g. mapping two hierarchies. You just need to register all concrete mappings and we will handle everything else.

public abstract class Message {}
public class TextMessage extends Message {}
public class ImageMessage extends Message {}
public class VideoMessage extends Message {}
public abstract class MessageDto {}
public class TextMessageDto extends MessageDto {}
public class ImageMessageDto extends MessageDto {}
public class VideoMessageDto extends MessageDto {}
@Mapper(componentModel = "spring")
public interface TextMessageMapper extends UnidirectionalMapper<TextMessage, TextMessageDto> {}

@Mapper(componentModel = "spring")
public interface ImageMessageMapper extends UnidirectionalMapper<ImageMessage, ImageMessageDto> {}

@Mapper(componentModel = "spring")
public interface VideoMessageMapper extends UnidirectionalMapper<VideoMessage, VideoMessageDto> {}
TextMessageDto textMessageDto = mappingFacade.map(new TextMessage(), TextMessageDto.class);
MessageDto imageMessageDto = mappingFacade.map(new ImageMessage(), MessageDto.class);
MessageDto videoMessageDto = mappingFacade.map(new VideoMessage(), MessageDto.class);

If MapStruct's default generated mappings are not enough, you can always write default implementation for UnidirectionalMapper's map method when registering a mapping. Or even use MapStruct's decorators for some specific cases.

Installation

Maven (pom.xml)

<dependencies>
    <dependency>
        <groupId>com.naharoo.commons</groupId>
        <artifactId>mapstruct-facade-spring-boot-starter</artifactId>
        <version>1.4.0</version>
    </dependency>
</dependencies>

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
                <source>${java.version}</source>
                <target>${java.version}</target>
                <annotationProcessorPaths>
                    <path>
                        <groupId>org.mapstruct</groupId>
                        <artifactId>mapstruct-processor</artifactId>
                        <version>1.4.2.Final</version>
                    </path>
                </annotationProcessorPaths>
            </configuration>
        </plugin>
    </plugins>
</build>

Gradle (build.gradle)

dependencies {
    annotationProcessor 'org.mapstruct:mapstruct-processor:1.4.2.Final'
    implementation 'com.naharoo.commons:mapstruct-facade-spring-boot-starter:1.4.0'
}

(Note: If you are using Lombok, then you should also include org.projectlombok:lombok-mapstruct-binding)

External Dependencies

This library has only two dependencies:

  1. org.springframework.boot:spring-boot-starter
  2. org.mapstruct:mapstruct. They will be promoted transitively. So their versions may be different from your application's existing dependencies'. In such cases, to avoid dependency conflicts you can adjust their versions in your local dependencyManagement configurations like so:

Maven

<dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct</artifactId>
            <version>1.3.0.Final</version>
        </dependency>
    </dependencies>
</dependencyManagement>

Gradle

dependencyManagement {
    dependencies {
        dependency("org.mapstruct:mapstruct:1.3.0.Final")
    }
}

(Note: If you are using Spring Boot BOM, or your application is extending Spring Boot Parent, then org.springframework.boot:spring-boot-starter dependency's version will be automatically adjusted).

License

Mapstruct Mapping Facade is Open Source software released under the Apache 2.0 license