これに似た質問を探していましたが、成功しませんでした。
私の目標: 特定のエンティティを作成した人のユーザー名を取得する必要があります。これは、Spring の @AuditTable を介して参照されるエンティティの監査テーブルに格納されます。
少し (できれば) 説明: この特定の状況では、org.springframework.data.jpa.repository.JpaSpecificationExecutor クラスと Predicate クラス (javax.persistence.criteria.Predicate) で構築された既存のクエリに追加しています。 . 結合は、Spring がテーブル名を取得するクラスの名前 (例: Join) で行われる傾向があります。私の質問は、Spring がエンティティのテーブル名ではなく、監査テーブル名で見えるようにする方法はありますか?
例は次のとおりです。
@Entity @Table(name = "persons")
@Audited @AuditTable(name = "audit_persons")
public class Person {
private String id;
private String name;
private Job job;
/*getters and setters*/
}
@Entity @Table(name = "jobs")
@Audited @AuditTable(name = "audit_jobs")
public class Job {
private String id;
private String name;
/*getters and setters*/
}
public final class PersonSpecification {
private PersonSpecification(){}
public static Specification<Person> findBy(final String attributeName, final Object filterValue) {
return new Specification<Person>() {
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
return cb.equal(root.get(attributeName), filterValue);
}
};
}
public static Specification<Person> findByJob(final Job job) {
return new Specification<Person>() {
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
Join<Person, Job> joinJob = root.join("job", JoinType.LEFT);
return cb.equal(joinJob.<Job>get("job"), job);
}
};
}
/*
public static Specification<Person> findByUsername(final String username) {
return new Specification<Person>() {
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
Join<Person, audit person?> join? = root.join(?, JoinType.LEFT);
return I don't know how to access;
}
};
}
*/
}
import static ().PersonSpecification.findBy;
import static ().PersonSpecification.findByJob;
@Service(value="personService")
public class PersonService {
@Autowired
private PersonDao personDao;
public List<Person> find(String id, String name, Job job /*, String username*/){
Specifications<Person> specifications = null;
if (!StringUtils.isBlank(id) && StringUtils.isNumeric(id)) {
specifications = loadFilter(specifications, findBy("id", id));
}else{
if (!StringUtils.isBlank(name)) {
specifications = loadFilter(specifications, findBy("name", name));
}
if (job != null) {
specifications = loadFilter(specifications, findByJob("job", job));
}
/*
if (!StringUtils.isBlank(username)) {
specifications = I don't know what to do here;
}
*/
}
return personDao.findAll(specifications);
}
}
考えられる解決策: 次のように、監査対象エンティティを参照する新しいエンティティを追加することを考えていました。
@Entity @Table(name = "audit_persons")
public class AuditedPerson {
private String id;
private Integer rev;
private Integer revtype;
private String username;
/*getters and setters*/
}
そして、私の方法は
public static Specification<Person> findByUsername(final String username) {
return new Specification<Person>() {
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
Join<Person, AuditedPerson> joinAudit = root.join(id, JoinType.LEFT);
return cb.and(cb.equal(joinAudit.get("username"), username),
cb.equal(joinAudit.get("revtype"), 0)); //this is so I get the creation instance's username
}
};
}
私はこれを機能させようとします。これを行う方法はすでにありますか?これを行うネイティブ クエリを追加する必要がありますか? ありがとう!