All of the source code for this tutorial can be found on here.
BuildScript
I'll start off with the build script. Pretty simple and straightforward. We are using Hibernate as our JPA provider, but any should work fine. For this tutorial I'm also using the in-memory database HSQLDB. Even in real projects I like to include HSQLDB for testing. It allows us to have unit tests that hit an actual database but are still fast and work if you aren't connected to a work VPN.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
apply plugin: "java" | |
apply plugin: "eclipse" | |
apply plugin: "idea" | |
sourceCompatibility = 1.7 | |
repositories { | |
mavenCentral() | |
} | |
ext { | |
hibernateVersion="4.3.1.Final" | |
springVersion="4.0.1.RELEASE" | |
} | |
dependencies { | |
compile "org.springframework:spring-context:$springVersion" | |
compile "org.springframework:spring-core:$springVersion" | |
compile "org.springframework:spring-jdbc:$springVersion" | |
compile "org.springframework:spring-orm:$springVersion" | |
compile "org.springframework:spring-tx:$springVersion" | |
compile 'org.springframework.data:spring-data-jpa:1.4.3.RELEASE' | |
compile "org.hibernate:hibernate-core:$hibernateVersion" | |
compile "org.hibernate:hibernate-entitymanager:$hibernateVersion" | |
compile 'org.hsqldb:hsqldb:2.3.0' | |
testCompile 'junit:junit:4.11' | |
testCompile "org.springframework:spring-test:$springVersion" | |
} |
Java Configuration
To enable the spring-data-jpa repositories add the annotation @EnableJpaRepositories and provide it with the package(s) that contain repositories. A couple of things to notice about this configuration file is that we tell Spring how to close the database when it is shutdown (@Bean(destroyMethod = "shutdown")). Failure to do this can lead to all kinds of weirdness. The second thing I wanted to point out are the properties at the bottom. Among some other things this will load the schema from the annotations in the Java files and create a file called schema.sql which you can use to create your database. This is super useful for testing using an in-memory database or as a base for creating your production database.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.example; | |
import java.util.Properties; | |
import javax.persistence.EntityManagerFactory; | |
import javax.sql.DataSource; | |
import org.hibernate.cfg.Environment; | |
import org.springframework.context.annotation.Bean; | |
import org.springframework.context.annotation.ComponentScan; | |
import org.springframework.context.annotation.Configuration; | |
import org.springframework.data.jpa.repository.config.EnableJpaRepositories; | |
import org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseBuilder; | |
import org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseType; | |
import org.springframework.orm.hibernate4.HibernateExceptionTranslator; | |
import org.springframework.orm.jpa.JpaTransactionManager; | |
import org.springframework.orm.jpa.JpaVendorAdapter; | |
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean; | |
import org.springframework.orm.jpa.vendor.Database; | |
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter; | |
import org.springframework.transaction.PlatformTransactionManager; | |
import org.springframework.transaction.annotation.EnableTransactionManagement; | |
@Configuration | |
@ComponentScan(basePackages = { "com.example.domain" }) | |
@EnableJpaRepositories(basePackages = { "com.example.repositories" }) | |
@EnableTransactionManagement | |
public class AppConfig { | |
public static final String DB_NAME = "testdb"; | |
@Bean(destroyMethod = "shutdown") | |
public DataSource dataSource() { | |
return new EmbeddedDatabaseBuilder().setType(EmbeddedDatabaseType.HSQL) | |
.setName(DB_NAME).build(); | |
} | |
@Bean(destroyMethod = "close") | |
public EntityManagerFactory entityManagerFactory() { | |
LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean(); | |
factory.setDataSource(dataSource()); | |
factory.setPersistenceUnitName(DB_NAME); | |
factory.setPackagesToScan("com.example.domain"); | |
factory.setJpaVendorAdapter(jpaAdapter()); | |
factory.setJpaProperties(jpaProperties()); | |
factory.afterPropertiesSet(); | |
return factory.getObject(); | |
} | |
@Bean | |
public JpaVendorAdapter jpaAdapter() { | |
HibernateJpaVendorAdapter adapter = new HibernateJpaVendorAdapter(); | |
adapter.setDatabase(Database.HSQL); | |
return adapter; | |
} | |
@Bean | |
public PlatformTransactionManager transactionManager() { | |
return new JpaTransactionManager(entityManagerFactory()); | |
} | |
@Bean | |
public HibernateExceptionTranslator exceptionTranslator() { | |
return new HibernateExceptionTranslator(); | |
} | |
public Properties jpaProperties() { | |
Properties properties = new Properties(); | |
properties.put(Environment.HBM2DDL_AUTO, "create"); | |
properties.put(Environment.HBM2DDL_IMPORT_FILES, "data.sql"); | |
properties.put( | |
"javax.persistence.schema-generation.create-database-schemas", "true"); | |
properties.put("javax.persistence.schema-generation.scripts.action", | |
"create"); | |
properties.put("javax.persistence.schema-generation.scripts.create-target", | |
"src/main/resources/schema.sql"); | |
properties.put("javax.persistence.database-product-name", "HSQL"); | |
return properties; | |
} | |
} |
The Domain and Repository
The domain is annotated like you've been doing using JPA or Hibernate. The real magic is the repository. It's just an interface. All the implementation details are hidden away in the spring-data-jpa project. For the majority of your CRUD methods you can use the methods provided or create your own using just the method signature. If you need more control you can use the @Query annotation and provide your own SQL.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.example.domain; | |
import java.io.Serializable; | |
import javax.persistence.Entity; | |
import javax.persistence.GeneratedValue; | |
import javax.persistence.Id; | |
import javax.persistence.Table; | |
@Entity | |
@Table | |
public class Person implements Serializable { | |
private static final long serialVersionUID = 1L; | |
@Id | |
@GeneratedValue | |
private Long id; | |
private String firstName; | |
private String lastName; | |
public Person() { | |
} | |
public Person(String firstName, String lastName) { | |
this.firstName = firstName; | |
this.lastName = lastName; | |
} | |
public Long getId() { | |
return id; | |
} | |
public void setId(Long id) { | |
this.id = id; | |
} | |
public String getFirstName() { | |
return firstName; | |
} | |
public void setFirstName(String firstName) { | |
this.firstName = firstName; | |
} | |
public String getLastName() { | |
return lastName; | |
} | |
public void setLastName(String lastName) { | |
this.lastName = lastName; | |
} | |
@Override | |
public int hashCode() { | |
final int prime = 31; | |
int result = 1; | |
result = prime * result + ((firstName == null) ? 0 : firstName.hashCode()); | |
result = prime * result + ((lastName == null) ? 0 : lastName.hashCode()); | |
return result; | |
} | |
@Override | |
public boolean equals(Object obj) { | |
if (this == obj) | |
return true; | |
if (obj == null) | |
return false; | |
if (getClass() != obj.getClass()) | |
return false; | |
Person other = (Person) obj; | |
if (firstName == null) { | |
if (other.firstName != null) | |
return false; | |
} | |
else if (!firstName.equals(other.firstName)) | |
return false; | |
if (lastName == null) { | |
if (other.lastName != null) | |
return false; | |
} | |
else if (!lastName.equals(other.lastName)) | |
return false; | |
return true; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.example.repositories; | |
import java.util.List; | |
import org.springframework.data.repository.CrudRepository; | |
import com.example.domain.Person; | |
public interface PersonRepository extends CrudRepository<Person, Long> { | |
List<Person> findByFirstName(String firstName); | |
List<Person> findByLastName(String lastName); | |
List<Person> findByLastNameContaining(String partialLastName); | |
} |
A test
A test to make sure everything works
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
insert into Person (firstName, lastName) values ('John', 'Doe'); | |
insert into Person (firstName, lastName) values ('John', 'Smith'); | |
insert into Person (firstName, lastName) values ('Jane', 'Doe'); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.example.repositories; | |
import static org.junit.Assert.assertEquals; | |
import static org.junit.Assert.assertNotNull; | |
import static org.junit.Assert.assertNull; | |
import static org.junit.Assert.assertTrue; | |
import java.util.List; | |
import javax.annotation.Resource; | |
import org.junit.Test; | |
import org.junit.runner.RunWith; | |
import org.springframework.test.context.ContextConfiguration; | |
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; | |
import com.example.AppConfig; | |
import com.example.domain.Person; | |
@RunWith(SpringJUnit4ClassRunner.class) | |
@ContextConfiguration(classes = { AppConfig.class }) | |
public class PersonRepositoryTest { | |
@Resource | |
private PersonRepository personRepository; | |
@Test | |
public void crud() { | |
Person me = new Person("Josh", "Brackett"); | |
Person saved = personRepository.save(me); | |
assertNotNull(saved); | |
assertNotNull(saved.getId()); | |
Long id = saved.getId(); | |
Person got = personRepository.findOne(id); | |
assertNotNull(got); | |
assertEquals(saved, got); | |
got.setFirstName("Joshua"); | |
Person updated = personRepository.save(got); | |
assertNotNull(updated); | |
assertEquals("Joshua", updated.getFirstName()); | |
personRepository.delete(id); | |
Person deleted = personRepository.findOne(id); | |
assertNull(deleted); | |
} | |
@Test | |
public void getJohns() { | |
List<Person> persons = personRepository.findByFirstName("John"); | |
assertNotNull(persons); | |
assertEquals(2, persons.size()); | |
for (Person person : persons) { | |
assertEquals("John", person.getFirstName()); | |
} | |
} | |
@Test | |
public void getDoes() { | |
List<Person> persons = personRepository.findByLastName("Doe"); | |
assertNotNull(persons); | |
assertEquals(2, persons.size()); | |
for (Person person : persons) { | |
assertEquals("Doe", person.getLastName()); | |
} | |
} | |
@Test | |
public void getPartialMatch() { | |
String partial = "mit"; | |
List<Person> persons = personRepository.findByLastNameContaining(partial); | |
assertNotNull(persons); | |
assertEquals(1, persons.size()); | |
for (Person person : persons) { | |
assertTrue(person.getLastName().contains(partial)); | |
} | |
} | |
} |