/anti-mapper

Simple helpers for Java object mapping

Primary LanguageJavaMIT LicenseMIT

anti-mapper

anti-mapper is a Java library, that assists the manual mapping of entities and DTOs.

In contrast to complex mapping frameworks, the anti-mapper comes with absolutely no reflection and guarantees zero magic. It offers no configuration at all and shines with a learing curve flat as a pancake.

Nevertheless, it is capable to boost your "I am faster doing it manually"-approach, because it helps you where it gets complicated: mapping sets, lists and maps. Additionally, it unifies your mapping efforts by offering consistent interfaces and base classes.

Introduction

DTOs and entities are part of a design pattern used in enterprise applications. Mapping these objects is a common practice. In Java, you have two possibilities to implement this: either do it manually or use one of the countless libraries.

  • Doing it manually is a boring task. It is always the same procedure but nevertheless the implementations fall apart and quickly become irregular, error-prone and hard to maintain.
  • Using a library simplifies and unifies the default cases, but usually makes special cases harder to solve. Most often you cannot simply implement theses special cases, you have to configure the mapper instead. And this can be a real pain.

anti-mapper offers a third possibility, the assisted manual mapping. That's why it is called anti-mapper. It is no classic mapping library - that's exactly the functionality it is missing.

Happy one: _"anti-mapper allows you implement the field-to-field mappings manually and just helps with the complex stuff like: collections, sets, lists and maps. It offers simple interfaces like a Transformer and a Merger which will keep your implementations uniform and consistent. There is no need for any configuration. Everything is pure Java code, checked by the compiler."

Grumpy one: "But OMG, what if I add a property to the entity and the DTO, I'm sure, I will forget the mapping. No reflection! What a stupid library!"

Happy one: "You will test your new property at least once, won't you? You will notice that the mapping is missing. If you don't, it's not the mappers fault."

Grumpy one: "I'm faster doing it manually."

Happy one: "You are mapping it manually! You will be fast, plus, it will be simple and consistent."

Grumpy one: "It will never map my lists the right way!"

Happy one: "It will. It uses a diff-algorithm for lists and reuses entities for Hibernate's sake. It can even handle deleted flags. There is no reason not to use the list mappings. But the best thing is: if you want to implement it in your style, it does not hinder you to do so."

Grumpy one: "It's not worth to learn it."

Happy one: "There is one simple recipe that fits all. Just read on."

Basic Concepts

DTO and Entity

anti-mapper commonly uses the terms entity and DTO.

What is called "an Entity"?

For anti-mapper an entity is an existing object that has to be modified when being mapped into. The most common case for this is a database entity, that's why it is called entity. In anti-mapper the process of mapping into an existing object called merging.

What is called "a DTO"?

A DTO on the other hand, is an object that will always be newly created when mapped into. The most common case for this is a DTO used in client-server applications. In anti-mapper the process of mapping into a newly created object is called transforming.

Transformer, Merger and Mapper

What is "transforming"?

The mapping of EntityDTO is performed by a Transformer. It maps an object (most often an entity) into a newly created object (most often a DTO).

What is "merging"?

The mapping of DTOEntity is performed by a Merger. It maps an object (most often a DTO) into an existing object (most often an entity).

What is "mapping"?

The mapping of EntityDTOEntity is performed by a Mapper. It is a combination of a Transformer and a Merger.

Hints

Hints are an array of objects that can be passed to transform and merge functions. Hints provide a context for mapping operations that make it simple to access default values or pass parent objects.

How-To

Transformer

If you want to map an object into a newly created object, then you implement a Transformer.

Create a new class and extend the AbstractTransformer. Make it a singleton - that means, you should always use the same instance for transforming. Write it thread-safe - that means, you class should only contain final fields!

Call the class something like *Mapper (even if it is just a Transformer). This is convenient and makes it easier to find and recognize mapper classes.

A Transformer is an interface with one method to implement:

  • DTO_TYPE transform(ENTITY_TYPE entity, Object... hints);

Additionally, it offers a lot of default methods for transforming streams, collections, sets, lists and maps.

The AbstractTransformer, you are using, encapsulates the null-checks. Usually, you just have to implement one method:

@Override
protected ParentDTO transformNonNull(ParentEntity entity, Object[] hints)
{

Start transforming by creating your target object. Final values are an easy task.

    ParentDTO dto = new ParentDTO(entity.getId());

If you want, you can add the DTO to the hints, just in case any child mapper needs it.

    hints = Hints.join(hints, dto);

Then set the values. Name and type conversions are trivial.

    dto.setName(entity.getKey());
    
    ZoneId timezone = Hints.hint(hints, ZoneId.class);
    dto.setTimestamp(entity.getTimestamp().toInstant().atZone(timezone).toLocalDateTime());

This is a call to a collective transform the Transformer interface provides. It transforms a collection into a grouped list.

    dto.setChilds(childMapper.transformToGroupedArrayLists(entity.getChilds(), child -> child.getType(), hints));

That's it, return the object.

    return dto;
}

Have a look at the ParentMapper-Sample!

Merger

If you want to map an object into an existing object, then you implement a Merger.

Create a new class and extend the AbstractMerger. Make it a singleton - that means, you should always use the same instance for transforming. Write it thread-safe - that means, you class should only contain final fields!

Call the class something like *Mapper (even if it is just a Merger). This is convenient and makes it easier to find and recognize mapper classes.

A Merger is an interface with two methods to implement:

  • ENTITY_TYPE merge(DTO_TYPE dto, ENTITY_TYPE entity, Object... hints);
  • boolean isUniqueKeyMatching(DTO_TYPE dto, ENTITY_TYPE entity, Object... hints);

Additionally, it offers a lot of default methods for merging streams, collections, sets, lists and maps.

The AbstractMerger, you are using, encapsulates the null-checks. Start by implementing the merge function:

@Override
protected ChildEntity mergeNonNull(ChildDTO dto, ChildEntity entity, Object[] hints)
{

Neither the DTO nor the entity is null. Set the values. Name and type conversions are trivial.

    entity.setKey(dto.getName());
    entity.setType(ChildType.valueOf(dto.getType()));

To set the parent, use the hints.

    entity.setParent(Hints.hint(hints, ParentEntity.class));

That's it, return the object.

    return entity;
}

If the entity is null, the AbstractMerger calls the create function before the merge. Implement it and set the final values.

@Override
protected ChildEntity create(ChildDTO dto, Object[] hints)
{
    return new ChildEntity(dto.getId());
}

The isUniqueKeyMatching method is used to find the right entity in collections and maps. The matching function usually just checks the ID and, if necessary, the unique keys.

@Override
public boolean isUniqueKeyMatching(ChildDTO dto, ChildEntity entity, Object... hints)
{
    return Objects.equals(dto.getId(), entity.getId());
}

And yes, this is enough, even if you map multiple DTOs with null as ID, because matched DTOs/Entities will not be matched twice.

Have a look at the ChildMapper-Sample!

Mapper

If you want a two-way mapping, then you implement a Mapper. It's the same as implementing a Transformer and a Merger in the same class.

That's all you need to know - as I said, flat as a pancake.

Best Practices

Mapping collections and maps

The Transformer and Merger interfaces contain methods for mapping collections and maps.

Start by using the transformAll and mergeAll methods, and specify the Iterable, Stream or Map you want to transform or merge. Those methods are all null-safe. You will get a StreamTransformer or StreamMerger, that contain methods to specify the target of the operation.

Watch out! The resulting maps and collections will not contain any null values, they will be filtered by default! If you need them, pass the Hint.KEEP_NULL hint.

If the resulting collection should be unmodifiable (e.g. for a cache), pass the Hint.UNMODIFIABLE hint.

You can also transform and merge groups. Use the transformGroup and mergeGroup methods for this. If your target is a group, you can use the toGroupedMap methods.

The merger basically distinguishes between ordered and mixed collections. Lists will keep the order when being mapped, sets will ignore the order. The methods for mixed collections just uses the isUniqueKeyMatching method, to find the matching entity, the methods for the ordered collection additionally use a diff algorithm, too. If you pass the Hint.UNMODIFIABLE to a merger, the maps and collections are assumed to be unmodifiable, and will always be new instances.

If you need to process all entities in the collection after merging, implement the afterMergeIntoCollection method.

isUniqueKeyMatching method

This method is used by the mapping method that map collections of DTOs to collections of entities. It helps to determine which DTO matches which entity. Despite it is just a little method, it can be a little bit tricky to implement.

First of all, the method will never be called with null values, this is checked beforehand.

If your objects have an unique id, the first step is to check for equality. If the ids are the same, return true.

if (Objects.equals(dto.getId(), entity.getId())) {
    return true;
} 

This even works if your have multiple DTOs with null as id (this usually happens when you add multiple new childs). Objects that have already been matched, will not be matched another time. There is a strict one-to-one relationship between DTOs and entities. If you return true for a DTO/entity pair, neither object will take part in another match.

Most of the time, checking the unique id is enough.

...
return false;     

If your objects have some functional keys you should check these, too. This is only necessary if you are not sure, that your DTO contains the id, even if the entity exists already.

if (Objects.equals(dto.getKey(), entity.getKey())) {
    return true;
}

If your matching method is wrong, the less bad thing that can happen, is that too many mappings will take place. This happens if you always return true. The worse thing may be, that entities will never be updated and any DTO will be added to the entities. This happens if you always return false.

If you have a unique key index in you database, it may happen, that inserting fails, if you don't check this key in the code. It is recommended, that you first check the id, if it is the same return true. Then check each property of the unique key and return false if any property differs.

afterMergeIntoCollection

Sometimes it is necessary to update all entities of a collection, after the merge has finished. You will need this, e.g. if you want to update an ordinal value.

The merger contains a method that is called after collection merges, this is a simple implementation that updates the ordinal:

public void afterMergeIntoCollection(Collection<ChildEntity> entities, Object... hints) {
    int ordinal = 0;
    for (ChildEntity entity : entities)
    {
        ordinal = Math.max(ordinal, entity.getOrdinal());
        entity.setOrdinal(ordinal++);
    }
}

Deleted-Flags and deleted-timestamps

To work with deleted-flags or deleted-timestamps, have a look at the DeletedFlag Sample.

It does not map entities that have been deleted, and sets the deleted-flag, if a DTO is missing.

You can find the trick in the ChildMapper:

@Override 
protected ChildDTO transformNonNull(ChildEntity entity, Object[] hints) 
{ 
    if (entity.isDeleted()) 
    { 
        return null; 
    } 
    return new ChildDTO(entity.getId(), entity.getName()); 
} 

@Override 
protected ChildEntity mergeNull(ChildEntity entity, Object[] hints) 
{ 
    entity.setDeleted(true); 
    return entity; 
}

Null values will automatically be filtered from the resulting collections.

Installation

Make sure you have Apache Maven. Just type:

mvn install