Java – use the builder in mapstruct (using the immutables annotation processor) to map objects to immutable objects

We use immutables framework to generate all DTOs Now we want to map these objects to another using mapstruct However, the generated dto is immutable. There is no setter or constructor, corresponding to the builder mode They are populated only by the corresponding builder accessed by the static Builder () – method

We tried to map dto1 to dto2 Builder, mapstruct will work if it can recognize setters in the builder, but these have no void return type, but return the builder itself for smooth connection

So this is the code for the example

We have two interfaces

@Value.Immutable
public interface MammalDto {
  public Integer getNumberOfLegs();
  public Long getNumberOfStomachs();
}

and

@Value.Immutable
public interface MammalEntity {
  public Long getNumberOfLegs();
  public Long getNumberOfStomachs();
}

Then we have mapper interface of mapstruct:

@Mapper(uses = ObjectFactory.class)
public interface SourceTargetMapper {
  SourceTargetMapper MAPPER = Mappers.getMapper( SourceTargetMapper.class );

  ImmutableMammalEntity.Builder toTarget(MammalDto source);
}

In order for maporg to find the builder, we need a factory:

public class ObjectFactory {

  public ImmutableMammalDto.Builder createMammalDto() {
    return ImmutableMammalDto.builder();
  }

  public ImmutableMammalEntity.Builder createMammalEntity() {
    return ImmutableMammalEntity.builder();
  }
}

To generate code, the compiler plug-in is instructed to use two annotation processors:

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.6.1</version>
    <configuration>
        <source>1.8</source>
        <target>1.8</target>
        <annotationProcessorPaths>
            <path>
                <groupId>org.immutables</groupId>
                <artifactId>value</artifactId>
                <version>2.2.8</version>
            </path>
            <path>
                <groupId>org.mapstruct</groupId>
                <artifactId>mapstruct-processor</artifactId>
                <version>1.2.0.Beta3</version>
            </path>
        </annotationProcessorPaths>
    </configuration>
</plugin>

Note: this only applies to mapstruct version > 1.2 x. The old version has a problem in MVN clean compile, that is, they can't find the source of immutable build In the second build (not clean), they will find immutables implementations because they are in the classpath before the annotation processor runs This bug has now been fixed

It's like a charm First, immutable implementations of interfaces are generated, and mapstruct uses them to generate builders

However, the test shows that the property is not set:

@Test
public void test() {
  MammalDto s = ImmutableMammalDto.builder().numberOfLegs(4).numberOfStomachs(3l).build();
  MammalEntity t = SourceTargetMapper.MAPPER.toTarget(s).build();
    assertThat(t.getNumberOfLegs()).isEqualTo(4);
    assertThat(t.getNumberOfStomachs()).isEqualTo(3);
}

Assertion failed When you look at the mapper generated by mapstruct, you will find that it obviously does not find any setters:

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",//...
)
public class SourceTargetMapperImpl implements SourceTargetMapper {
    private final ObjectFactory objectFactory = new ObjectFactory();

    @Override
    public Builder toTarget(MammalDto source) {
        if ( source == null ) {
            return null;
        }

        Builder builder = objectFactory.createMammalEntity();
        return builder;
    }
}

Returns an empty builder I think the reason is the setter implementation of the generated builder, because it returns itself to create a smooth API:

public final Builder numberOfLegs(Long numberOfLegs) {
  this.numberOfLegs = Objects.requireNonNull(numberOfLegs,"numberOfLegs");
  return this;
}

Is there any way for mapstruct to find these setters? Or even a better way to handle these immutable objects with a builder?

Editor: as I said in my comments, I met issue #782 In version 1.2 0. Beta 3 is still not supported by the builder But there are several discussions on this topic, so if one encounters the same problem, he may be interested

Solution

Our project has the same problem

You can also try It's better to use builders and object factories directly

@ Value. Modifiable uses setters to generate implementations

@ Value. Style (create = "new") generates the public no args constructor

@Value.Immutable
@Value.Modifiable
@Value.Style(create = "new")
public interface MammalEntity {
    public Long getNumberOfLegs();
    public Long getNumberOfStomachs();
}

Then your mapper will be simpler and does not need to be in an object factory

@Mapper
public interface SourceTargetMapper {

  ModifiableMammalEntity toTarget(MammalDto source);
}

In this case, mapstruct can see the setter in modifiable mammalentity

The usage of this mapper looks like

// Here you don't need to worry about implementation of MammalEntity is. The interface `MammalEntity` is immutable.
MammalEntity mammalEntity = sourceTargetMapper.toTarget(source);
The content of this article comes from the network collection of netizens. It is used as a learning reference. The copyright belongs to the original author.
THE END
分享
二维码
< <上一篇
下一篇>>