1

我有一个典型的例子,一个帖子有很多标签,一个标签有很多帖子。我没有使用典型的@ManyToMany,而是在中间使用了一个名为 TAGPOST 的域对象,它还允许我在其中拥有有用的数据,例如何时使用给定标签标记帖子等。每个 POST 和 TAG 对应,@OneToMany与 TAGPOST有关系。

具体要求是一个帖子不能包含两次相同的标签,因此 TAGPOST.post 和 TAGPOST.tag 对必须始终是唯一的。通常,我会在表中创建一个复合主键对,负责存储 TAGPOST 对象。

AFAIK,没有办法表达这种独特的约束。我已标记jpa.ddl=update,这意味着每次我将应用程序移动到新环境时,我都必须在数据库中手动修复它。这非常不方便,而且容易出错,尤其是在单元测试时,因为在每次迭代中都会或多或少地创建和删除数据库。

我什至在考虑手动进行检查@PrePersist,甚至将检查移动到业务层中,比如创建一个 PostService。

我该怎么办?我错过了 Play 默认拥有的东西吗?一些巧妙的注释来表达@ManyToOneTAGPOST 类的属性的唯一性?

仅供参考:我正在使用 Play 1.2.5

编辑: TAGPOST 类如下所示:

@Entity
public class TagPost extends Model {

    @ManyToOne
    public Tag tag;

    @ManyToOne
    public Post post;

    public Date dateAdded;

    ...
}
4

1 回答 1

0

Check我为数据库唯一性编写了一个自定义。也许你应该定制它。

DBUnique.java

package models.check;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import net.sf.oval.configuration.annotation.Constraint;
import play.db.jpa.GenericModel;

@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.FIELD, ElementType.PARAMETER })
@Constraint(checkWith = DbUniqueCheck.class)
public @interface DBUnique {
    String message() default DbUniqueCheck.mes;

    Class<? extends GenericModel> modelClass();

    String field() default ""; // field name will be used
}

DbUniqueCheck.java

package models.check;


import net.sf.oval.Validator;
import net.sf.oval.configuration.annotation.AbstractAnnotationCheck;
import net.sf.oval.context.FieldContext;
import net.sf.oval.context.OValContext;

import org.apache.commons.lang.StringUtils;

import play.db.jpa.GenericModel.JPAQuery;

@SuppressWarnings("serial")
public class DbUniqueCheck extends AbstractAnnotationCheck<DBUnique> {

final static String mes = "validation.dbunique";
DBUnique dbUnique;

@Override
public void configure(DBUnique dBUnique) {
    this.dbUnique = dBUnique;
    setMessage(dBUnique.message());
}

public boolean isSatisfied(Object validatedObject, Object value, OValContext context, Validator validator) {
    try {
        String field = dbUnique.field();
        if (field == null || field.isEmpty()) {
            field = ((FieldContext) context).getField().getName();
        }
        JPAQuery q = (JPAQuery) dbUnique.modelClass().getMethod("find", String.class, Object[].class)
                .invoke(null, "by" + StringUtils.capitalize(field), new Object[] { value });
        return q.first() == null;
    } catch (Exception e) {
        e.printStackTrace();
    }
    return false;
}

}

用法:链接到要点

它只是检查给定实例的给fieldclass实例在 db 中是否唯一。也许你应该做这样的东西..

于 2013-03-23T08:37:00.727 回答