Lui*_*ang 7 java spring-data spring-data-jpa spring-data-rest
我是 Spring Boot 的初学者,无法解决问题。我有一个实体类 (Customer) 和一个 REST 存储库 (CustomerRepository)。该类包含一些我不希望 REST 存储库公开的敏感字段。因此,我使用 @JsonIgnore 注解对这些字段进行注解,如下所示:
package br.univali.sapi.entities;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Inheritance;
import javax.persistence.InheritanceType;
import com.fasterxml.jackson.annotation.JsonIgnore;
@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
public class Customer
{
@Id
@GeneratedValue
private Long id = null;
private String login;
private String name;
@JsonIgnore
private String password;
@JsonIgnore
private String email;
public Customer()
{
}
public Long getId()
{
return id;
}
public void setId(Long id)
{
this.id = id;
}
public String getLogin()
{
return login;
}
public void setLogin(String login)
{
this.login = login;
}
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
public String getPassword()
{
return password;
}
public void setPassword(String password)
{
this.password = password;
}
public String getEmail()
{
return email;
}
public void setEmail(String email)
{
this.email = email;
}
}
Run Code Online (Sandbox Code Playgroud)
一切工作正常,我的 REST API 返回了所需的结果。但是,当我向 API 发出 POST 请求以插入新实体时,我收到数据库错误:"column password can't be null", "column email can't be null"。
密码和电子邮件与其他参数一起在 POST 请求中发送到服务器,但似乎被忽略。如果我删除 @JsonIgnore 注释,实体将正常保留。
我知道我可以使用投影来隐藏这些字段。但投影是 URL 中的可选参数。这样,有经验的用户就能够从请求 URL 中删除参数并无论如何都能看到这些字段。
如果我可以隐式强制执行投影,那就可以解决问题,但这似乎无法仅使用 Spring 框架来完成。我可以使用 Apache URL 重写来实现这一点,但维护工作会很糟糕。
有人知道我该如何解决这个问题吗?谢谢!
编辑1:
我相信我找到了使用 DTO/投影(数据传输对象)的解决方案。您必须创建两个 DTO,一个用于显示实体,另一个用于更新实体,如下所示:
public interface CustomerViewDTO
{
public Long getId();
public String getLogin();
public String getName();
}
Run Code Online (Sandbox Code Playgroud)
public class CustomerUpdateDTO
{
private String login;
private String name;
private String password;
private String email;
// Getters and Setters ommited for breviety
}
Run Code Online (Sandbox Code Playgroud)
现在,在存储库上使用 DTO,Spring 将发挥它的魔力:
@Transactional(readOnly = true)
public interface CustomerRepository extends JPARepository<Customer, Long>
{
// Using derived query
public CustomerViewDTO findByIdAsDTO(Long id);
// Using @Query annotation
@Query("SELECT C FROM Customer C WHERE C.id = :customerId")
public CustomerViewDTO findByIdAsDTO(@Param("customerId") Long id);
}
Run Code Online (Sandbox Code Playgroud)
为了进行更新,您会在控制器上收到 DTO 并将其映射到服务上的实体,如下所示:
@RestController
public class CustomerController
{
@Autowired
private CustomerService customerService;
@RequestMapping(method = "PATCH", path = "/customers/{customerId}")
public ResponseEntity<?> updateCustomer(@PathVariable Long customerId, @RequestBody CustomerUpdateDTO customerDTO)
{
CustomerViewDTO updatedCustomer = customerService.updateCustomer(customerId, customerDTO);
return ResponseEntity.ok(updatedCustomer);
}
@RequestMapping(method = GET, path = "/customers/{customerId}")
public ResponseEntity<?> findCustomerById(@PathVariable Long customerId)
{
return ResponseEntity.ok(customerService.findByIdAsDTO(Long));
}
}
Run Code Online (Sandbox Code Playgroud)
@Service
public class CustomerService
{
@Autowired
private CustomerRepository customerRepository;
// Follow this tutorial:
// https://www.baeldung.com/entity-to-and-from-dto-for-a-java-spring-application
@Autowired
private ModelMapper modelMapper;
@Transactional(readOnly = false)
public CustomerViewDTO updateCustomer(Long customerId, CustomerUpdateDTO customerDTO)
{
Customer customerEntity = customerRepository.findById(customerId);
// This copies all values from the DTO to the entity
modelMapper.map(customerDTO, customerEntity);
// Now we have two aproaches:
// 1. save the entity and convert back to DTO manually using the model mapper
// 2. save the entity and call the repository method which will convert to the DTO automatically
// The second approach is the one I use for several reasons that
// I won't explain here
// Here we use save and flush to force JPA to execute the update query. This way, when we do the select the entity will come with the fields updated
customerEntity = customerRepository.saveAndFlush(customerEntity);
// First aproach
return modelMapper.map(customerEntity, CustomerViewDTO.class);
// Second approach
return customerRepository.findByIdAsDTO(customerId);
}
@Transactional(readOnly = true)
public CustomerViewDTO findByIdAsDTO(Long customerId)
{
return customerRepository.findByIdAsDTO(customerId);
}
}
Run Code Online (Sandbox Code Playgroud)
我想你对数据库级别没有空约束。基本上,当您@JsonIgnore在字段上执行操作时,您只是没有传递它并且数据库无法插入该值,这是非常清楚的。
所以我在这里看到的解决方案如下:
1)删除@JsonIgnore注释以便能够执行POST请求。
2) 使用时使用投影GET。您可以配置Spring Data REST为始终使用投影,但默认情况下Spring Data REST始终返回完整的JSON单个资源(在您的情况下使用电子邮件和密码),这对我来说很奇怪。这个问题的解决方案我在另一个答案中写过,因为我也遇到了同样的问题。只需ProjectingProcessor为所有投影使用一些默认名称即可。不要忘记使用方法在您的配置中添加投影config.getProjectionConfiguration().addProjection 。
3) 避免步骤 2 的唯一方法是禁止级别GET上的单个资源Spring Data REST并使用自定义控制器。但我会采用步骤 2 中的解决方案以避免样板控制器。
| 归档时间: |
|
| 查看次数: |
3191 次 |
| 最近记录: |