У нас самая простая задача CRUD с JPA 1.0 и JAX-WS.
Допустим, у нас есть сущность Person.
@Entity
public class Person
{
@Id
private String email;
@OneToOne(fetch = FetchType.LAZY)
@JoinColumn(insertable = false, updatable = false)
private ReadOnly readOnly;
@Column
private String name;
@XmlElement
public String getEmail()
{
return email;
}
public void setEmail(String email)
{
this.email = email;
}
@XmlElement
public Long getReadOnlyValue()
{
return readOnly.getValue();
}
// more get and set methods
}
Вот сценарий.
Клиент делает запрос веб-службы для создания лица. На стороне сервера все просто.
И это работает как ожидалось.
@Stateless
@WebService
public class PersonService
{
@PersistenceContext(name = "unit-name")
private EntityManager entityManager;
public Person create(Person person)
{
entityManager.persist(person);
return person;
}
}
Теперь клиент пытается обновить пользователя, и именно здесь, как по мне, JPA показывает свою несостоятельность.
public Person update(Person person)
{
Person existingPerson = entityManager.find(Person.class, person.getEmail());
// some logic with existingPerson
// ...
// At this point existingPerson.readOnly is not null and it can't be null
// due to the database.
// The field is not updatable.
// Person object has readOnly field equal to null as it was not passed
// via SOAP request.
// And now we do merge.
entityManager.merge(person);
// At this point existingPerson.getReadOnlyValue()
// will throw NullPointerException.
// And it throws during marshalling.
// It is because now existingPerson.readOnly == person.readOnly and thus null.
// But it won't affect database anyhow because of (updatable = false)
return existingPerson;
}
Чтобы избежать этой проблемы, мне нужно предоставить set для объекта readOnly и сделать что-то подобное перед слиянием.
Person existingPerson = entityManager.find(Person.class, person.getEmail());
person.setReadOnlyObject(existingPerson.getReadOnlyObject()); // Arghhh!
Мои вопросы:
- Это функция или просто
несогласованность?
- Как ты (или будешь
вы) справляетесь с такими ситуациями? пожалуйста
не советуйте мне использовать DTO.