数据模型的首次迭代
接下来我们要开始完成我们的博客引擎的模型部分。
JPA入门
模型层是一个Play应用的核心(对于其他Web框架也同样成立)。它是一个对应用操作的资源的领域特定的表示。因为我们想要创建一个博客引擎,模型层就包括User,Post和Comment(用户,博文和评论)。
因为大多数模型对象需要在应用停止运行时保留下来,我们需要把它们存储在持久性数据库中。一个普遍的选择是使用关系型数据库。因为Java是一个面向对象的语言,我们将使用一个ORM来减少一些繁琐的工作。
JPA是一个给ORM定义一套标准API的Java规范。作为一个JPA的实现,Play使用猿媛皆知的Hibernate框架。之所以使用JPA而不是原生的Hibernate API,是因为这样所有的映射都可以用Java对象直接完成。
如果之前用过Hibernate或JPA,你将惊讶于Play所添加的包装。不再需要配置什么了;JPA与Play框架合一。
如果你不知道JPA,你可以在继续之前阅读一些JPA实现的介绍
User类
我们首先来完成User类。创建新文件/yabe/app/models/User.java
,并写入下面的内容:
package models;
import java.util.*;
import javax.persistence.*;
import play.db.jpa.*;
@Entity
public class User extends Model {
public String email;
public String password;
public String fullname;
public boolean isAdmin;
public User(String email, String password, String fullname) {
this.email = email;
this.password = password;
this.fullname = fullname;
}
}
@Entity
注解(annotation)标记该类成为托管的JPA实体(managed JPA Entity),而Model父类将自动提供一些接下来将会用到的有用的JPA辅助函数。这个类的所有成员变量都会被持久化到数据库中。
默认情况下,对应的表就是'User'。如果想要使用一个'user'是保留关键字的数据库,你需要给JPA映射指定一个不同的表名。要想这么做,使用
@Table(name="blog_user")
注解User
类。
你的模型对象不一定得继承自
play.db.jpa.Model
类。你也可以使用原生JPA。但继承自该类往往是个更好的选择,因为它使得运用JPA变得更为简单。
如果之前用过JPA,你知道每个JPA实体都需要提供一个@Id属性。在这里,Model父类已经提供了一个自动生成的ID,在大多数情况下,这样就行了。
不要认为生成的id成员变量是函数变量(functional identifier),其实它是技术变量(technical identifier)。区分这两概念通常是个好主意,记住自动生成的ID是一个技术变量(译注:这里我弄不懂,以下附上原文)
Don’t think about this provided id field as a functional identifier but as a technical identifier. It is generally a good idea to keep both concepts separated and to keep an automatically generated numeric ID as a technical identifier.
如果你写过Java,心中可能已经敲起了警钟,因为我们居然大量使用公有成员!在Java(一如其他面向对象语言),最佳实践通常是尽量保持各成员私有,并提供getter和setter。这就是封装,面向对象设计的基本概念之一。事实上,Play已经考虑到这一点,在自动生成getter和setter的同时保持封装;等下我们将看到它是怎么做到的。
现在你可以刷新主页面,看一下结果。当然,除非你犯错,否则应该什么变化都看不到:D。Play自动编译并加载了User类,不过这没有给应用添加任何新特性。
写下第一个测试
测试新增的User类的一个好方法是写下JUnit测试用例。它会允许你增量开发的同时保证一切安好。
要运行一个测试用例,你需要在'test'模式下运行应用。停止当前正在运行的应用,打开命令行并输入:
~$ play test
play test
命令就像play run
,不过它加载的是一个测试运行器模块,使得你可以直接在浏览器中运行测试套件。
当你在
test mode
中运行Play应用时,Play会自动切换到test
框架ID并加载对应的application.conf
。阅读框架ID文档来了解更多。
在浏览器打开http://localhost:9000/@tests页面来看看测试运行器。尝试选择所有的默认测试并运行;应该全部都会是绿色……但是默认的测试其实什么都没测:D
我们将使用JUnit测试来测试模型部分。如你所见,已经存在一个默认的BasicTests.java
,所以让我们打开它(/yabe/test/BasicTest.java
):
import org.junit.*;
import play.test.*;
import models.*;
public class BasicTest extends UnitTest {
@Test
public void aVeryImportantThingToTest() {
assertEquals(2, 1 + 1);
}
}
删除没用的默认测试(aVeryImportantThingToTest
),创建一个注册新用户并进行检查的测试:
@Test
public void createAndRetrieveUser() {
// Create a new user and save it
new User("bob@gmail.com", "secret", "Bob").save();
// Retrieve the user with e-mail address bob@gmail.com
User bob = User.find("byEmail", "bob@gmail.com").first();
// Test
assertNotNull(bob);
assertEquals("Bob", bob.fullname);
}
如你所见,Model父类给我们提供了两个非常有用的方法:save()
和find()
。
你可以在Play文档中的JPA支持阅读到Model类的更多方法。
在test runner中选择BasicTests.java
,点击开始,看一下是不是全都变绿了。
我们将需要在User类中添加一个方法,来检查给用户的用户名和密码是否存在了。让我们完成它,并且测试它。
在User.java
中,添加connect()
方法:
public static User connect(String email, String password) {
return find("byEmailAndPassword", email, password).first();
}
如今测试用例成这样:
@Test
public void tryConnectAsUser() {
// Create a new user and save it
new User("bob@gmail.com", "secret", "Bob").save();
// Test
assertNotNull(User.connect("bob@gmail.com", "secret"));
assertNull(User.connect("bob@gmail.com", "badpassword"));
assertNull(User.connect("tom@gmail.com", "secret"));
}
每次修改之后,你都可以从Play测试运行器运行所有的测试,来确保没有什么被破坏了。
Post类
Post
类表示博客文章。让我们写下代码:
package models;
import java.util.*;
import javax.persistence.*;
import play.db.jpa.*;
@Entity
public class Post extends Model {
public String title;
public Date postedAt;
@Lob
public String content;
@ManyToOne
public User author;
public Post(User author, String title, String content) {
this.author = author;
this.title = title;
this.content = content;
this.postedAt = new Date();
}
}
这里我们使用@Lob
注解告诉JPA来使用字符大对象类型(clob)来存储文章内容。我们也声明跟User
类的关系是@ManyToOne
。这意味着每个Post
对应一个User
,而每个User
可以有多个Post
。
PostgreSQL的最近版本不会将
@Lob
注解的String
成员存储成字符大对象类型,除非你额外用@Type(type = "org.hibernate.type.TextType")
注解该成员。
我们将写一个新的测试用例来检查Post
类能否正常工作。但在写下更多测试之前,我们需要修改下JUnit测试类。在当前测试中,数据库的内容永不删除,所以每次运行测试都会创建越来越多的对象。假如将来我们需要测试对象的数目是否正确,这将会是一个问题。
所以先写一个JUnit的setup()
方法在每次测试之前清空数据库:
public class BasicTest extends UnitTest {
@Before
public void setup() {
Fixtures.deleteDatabase();
}
…
}
@Before是JUnit测试工具的一个核心概念
如你所见,Fixtures
类是一个在测试时帮助处理数据库的类。再次运行测试并检查是否一切安好。之后接着下下一个测试:
@Test
public void createPost() {
// Create a new user and save it
User bob = new User("bob@gmail.com", "secret", "Bob").save();
// Create a new post
new Post(bob, "My first post", "Hello world").save();
// Test that the post has been created
assertEquals(1, Post.count());
// Retrieve all posts created by Bob
List<Post> bobPosts = Post.find("byAuthor", bob).fetch();
// Tests
assertEquals(1, bobPosts.size());
Post firstPost = bobPosts.get(0);
assertNotNull(firstPost);
assertEquals(bob, firstPost.author);
assertEquals("My first post", firstPost.title);
assertEquals("Hello world", firstPost.content);
assertNotNull(firstPost.postedAt);
}
不要忘记导入
java.util.List
,否则你会得到一个编译错误。
添加Comment类
最后,我们需要给博文添加评论功能。
创建Comment
类的方式十分简单直白。
package models;
import java.util.*;
import javax.persistence.*;
import play.db.jpa.*;
@Entity
public class Comment extends Model {
public String author;
public Date postedAt;
@Lob
public String content;
@ManyToOne
public Post post;
public Comment(Post post, String author, String content) {
this.post = post;
this.author = author;
this.content = content;
this.postedAt = new Date();
}
}
让我们写下第一个测试用例:
@Test
public void postComments() {
// Create a new user and save it
User bob = new User("bob@gmail.com", "secret", "Bob").save();
// Create a new post
Post bobPost = new Post(bob, "My first post", "Hello world").save();
// Post a first comment
new Comment(bobPost, "Jeff", "Nice post").save();
new Comment(bobPost, "Tom", "I knew that !").save();
// Retrieve all comments
List<Comment> bobPostComments = Comment.find("byPost", bobPost).fetch();
// Tests
assertEquals(2, bobPostComments.size());
Comment firstComment = bobPostComments.get(0);
assertNotNull(firstComment);
assertEquals("Jeff", firstComment.author);
assertEquals("Nice post", firstComment.content);
assertNotNull(firstComment.postedAt);
Comment secondComment = bobPostComments.get(1);
assertNotNull(secondComment);
assertEquals("Tom", secondComment.author);
assertEquals("I knew that !", secondComment.content);
assertNotNull(secondComment.postedAt);
}
你可以看到Post和Comments之间的联系并不紧密:我们不得不通过查询来获得所有跟某一个Post
关联的评论。通过在Post
和Comment
类之间建立新的关系,我们可以改善这一点。
在Post
类添加comments
成员:
...
@OneToMany(mappedBy="post", cascade=CascadeType.ALL)
public List<Comment> comments;
public Post(User author, String title, String content) {
this.comments = new ArrayList<Comment>();
this.author = author;
this.title = title;
this.content = content;
this.postedAt = new Date();
}
...
注意现在我们用mappedBy
属性来告诉JPAComment
类的post成员是维持这个关系的一方。当你用JPA定义一个双向关系时,需要指定哪一方来维持这个关系。在这个例子中,因为Comment
示例依赖于Post
,我们按Comment.post
的反向来定义关系。
我们也设置了cascade
属性来告诉JPA,我们希望Post
的删除将级联影响到comments
。也即是,如果你删除一个博文时,所有相关的评论也将一并删除。
由于有了这个新关系,我们可以给Post
类添加一个辅助方法来简化评论的添加:
public Post addComment(String author, String content) {
Comment newComment = new Comment(this, author, content).save();
this.comments.add(newComment);
this.save();
return this;
}
让我们写多一个测试检查它能否工作:
@Test
public void useTheCommentsRelation() {
// Create a new user and save it
User bob = new User("bob@gmail.com", "secret", "Bob").save();
// Create a new post
Post bobPost = new Post(bob, "My first post", "Hello world").save();
// Post a first comment
bobPost.addComment("Jeff", "Nice post");
bobPost.addComment("Tom", "I knew that !");
// Count things
assertEquals(1, User.count());
assertEquals(1, Post.count());
assertEquals(2, Comment.count());
// Retrieve Bob's post
bobPost = Post.find("byAuthor", bob).first();
assertNotNull(bobPost);
// Navigate to comments
assertEquals(2, bobPost.comments.size());
assertEquals("Jeff", bobPost.comments.get(0).author);
// Delete the post
bobPost.delete();
// Check that all comments have been deleted
assertEquals(1, User.count());
assertEquals(0, Post.count());
assertEquals(0, Comment.count());
}
这次全绿了么?
使用Fixtures来写更复杂的测试
当你开始写更加复杂的测试,你通常需要一些测试数据。Fixtures允许你在一个YAML文件中描述你的模型,并在测试开始前加载。
编辑/yabe/test/data.yml
并开始描述一个User:
User(bob):
email: bob@gmail.com
password: secret
fullname: Bob
...
呃,因为data.yml
有点大,你可以在这里下载它。
现在我们可以创建一个加载数据并对它运行一些断言的测试用例:
@Test
public void fullTest() {
Fixtures.loadModels("data.yml");
// Count things
assertEquals(2, User.count());
assertEquals(3, Post.count());
assertEquals(3, Comment.count());
// Try to connect as users
assertNotNull(User.connect("bob@gmail.com", "secret"));
assertNotNull(User.connect("jeff@gmail.com", "secret"));
assertNull(User.connect("jeff@gmail.com", "badpassword"));
assertNull(User.connect("tom@gmail.com", "secret"));
// Find all of Bob's posts
List<Post> bobPosts = Post.find("author.email", "bob@gmail.com").fetch();
assertEquals(2, bobPosts.size());
// Find all comments related to Bob's posts
List<Comment> bobComments = Comment.find("post.author.email", "bob@gmail.com").fetch();
assertEquals(3, bobComments.size());
// Find the most recent post
Post frontPost = Post.find("order by postedAt desc").first();
assertNotNull(frontPost);
assertEquals("About the model layer", frontPost.title);
// Check that this post has two comments
assertEquals(2, frontPost.comments.size());
// Post a new comment
frontPost.addComment("Jim", "Hello guys");
assertEquals(3, frontPost.comments.size());
assertEquals(4, Comment.count());
}
你可以在YAML manual page中阅读更多关于Play和YAML的内容。
保存你的成果
现在我们已经完成了博客引擎的大部分模型层。既然已经创建并测试好了模型层,我们可以开始开发这个Web应用了。
不过在继续前进之前,是时候用Bazaar保存你的成果。打开命令行,输入bzr st
来看看在前一个提交之后做的修改:
$ bzr st
如你所见,一些新文件不在版本控制之中。test-result
目录不需要加入到版本控制,所以就忽略它。
$ bzr ignore test-result
通过bzr add
向版本控制加入其他文件。
$ bzr add
你现在可以提交你的改动了。
$ bzr commit -m "The model layer is ready"