We are a small team starting a new project, and developing the back-end using Spring-Boot framework.
The back-end project will have the following standard layers of a Spring-Boot application:
- Controller (map the endpoints and handle)
- Service (to handle business logic)
- Repository (to abstract and interact with the database)
We also have an entity package where all the classes representing database entities are located. We also use a library to map entities with Api models.
It did not last long and our first debate within the team occurred. So I thought it's a good idea to ask for advice out there.
My other partners think that the controllers should contain as few lines of code as possible. Ideally containing only one line of code, where the respective Service is called, and the rest is handled in the Service.
I am against that approach, because if we would do that, the Service and the Controller would end up having the same interface, which feels wrong.
However, I agree with the principle that the controllers should have as few lines of code as possible. I think that the controllers methods should accept and return only Api models, and no concept of entities should be present in the controllers.
At the same time, I think that the services method should accept and return entities, and not work with ApiModels.
Should we introduce a new layer between Controller and Service? Is there some standard, for which I am not aware, related to this topic?
I am going to give you a concrete example with two approaches (mine and my partners'):
First approach:
public class UserController implements UsersApi {
@Autowired
private UserService userService;
@Override
public ResponseEntity<UserAPIModel> createUser(@Valid UserAPIModel body) {
User createdUser = userService.save(UserMapper.INSTANCE.toUserEntity(body));
return ResponseEntity.ok(UserMapper.INSTANCE.toUserAPIModel(createdUser));
}
}
And the service:
public class UserService {
@Autowired
private UserRepository repository;
public User save(User entity) {
return repository.save(entity);
}
}
The second approach:
public class UserController implements UsersApi {
@Autowired
private UserService userService;
@Override
public ResponseEntity<UserAPIModel> createUser(@Valid UserAPIModel body) {
return ResponseEntity.ok(userService.createUser(body));
}
}
and the service:
public class UserService {
@Autowired
private UserRepository repository;
public UserAPIModel createUser(UserAPIModel body) {
User user = repository.save(UserMapper.INSTANCE.toUserEntity(body));
return UserMapper.INSTANCE.toUserAPIModel(user);
}
}
As you can see in the first approach, the mapping from entity to Api Model and vice versa is done in the Controller. In the second approach this mapping is done in the Service.
Which one would you recommend? Or do you think it's better to introduce a new layer between Controller and Service?
Making a decision on this simple example, can help us to create a more general rule and set a standard for similar problems in the future.
Ho guys!
I too started working on Springboot a few months ago, I can tell you that from personal experience it is better to create a Mapper between Controller and Service in order to proceed better. Generally, especially in this type of projects, it is imperative to diversify and not to make the Controller or the Service do tasks outside of their use. The controller in this case you only need to start a command, there shouldn't be any kind of direct connections to Repo either. from experience I tell you that in larger projects this mentality pays off