Java Heroku上带Spring Boot的Postgresql存在问题
我决定在Heroku上试用Spring Boot,一切都很好…除了数据库之外 我尝试了很多不同的方法来让它工作,遵循了我在网上找到的很多不同的方法,也从StackOverflow上的问题中找到了很多不同的方法,但似乎没有什么对我有效。 我将只发布我试图部署的应用程序的最低限度 下面是desployed罚款,但当我试图张贴的东西,我得到以下回应:Java Heroku上带Spring Boot的Postgresql存在问题,java,postgresql,heroku,spring-boot,spring-data-jpa,Java,Postgresql,Heroku,Spring Boot,Spring Data Jpa,我决定在Heroku上试用Spring Boot,一切都很好…除了数据库之外 我尝试了很多不同的方法来让它工作,遵循了我在网上找到的很多不同的方法,也从StackOverflow上的问题中找到了很多不同的方法,但似乎没有什么对我有效。 我将只发布我试图部署的应用程序的最低限度 下面是desployed罚款,但当我试图张贴的东西,我得到以下回应: {“时间戳”:1413600470146,“错误”:“不支持的媒体类型”,“状态”:415,“消息”:“不支持的媒体类型”} 当我试图获取时,Herok
{“时间戳”:1413600470146,“错误”:“不支持的媒体类型”,“状态”:415,“消息”:“不支持的媒体类型”}
当我试图获取时,Heroku日志中出现以下异常:
嵌套异常为org.hibernate.exception.sqlgrammareexception:无法提取具有根本原因的结果集]
org.postgresql.util.PSQLException: ERROR: relation "exampleEntity" does not exist
Position: 109
at org.postgresql.core.v3.QueryExecutorImpl.receiveErrorResponse(QueryExecutorImpl.java:2103)
at org.postgresql.core.v3.QueryExecutorImpl.processResults(QueryExecutorImpl.java:1836)
at org.postgresql.core.v3.QueryExecutorImpl.execute(QueryExecutorImpl.java:257)
at org.postgresql.jdbc2.AbstractJdbc2Statement.execute(AbstractJdbc2Statement.java:512)
at org.postgresql.jdbc2.AbstractJdbc2Statement.executeWithFlags(AbstractJdbc2Statement.java:388)
at org.postgresql.jdbc2.AbstractJdbc2Statement.executeQuery(AbstractJdbc2Statement.java:273)
at org.apache.commons.dbcp.DelegatingPreparedStatement.executeQuery(DelegatingPreparedStatement.java:96)
....
因此,我有以下pom.xml:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>project-name</artifactId>
<version>0.0.1-SNAPSHOT</version>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.0.2.RELEASE</version>
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>1.7</maven.compiler.source>
<maven.compiler.target>1.7</maven.compiler.target>
<jersey.version>2.8</jersey.version>
</properties>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.glassfish.jersey</groupId>
<artifactId>jersey-bom</artifactId>
<version>${jersey.version}</version>
<type>pom</type>
<scope>import</scope>
</dependency>
</dependencies>
</dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>9.1-901-1.jdbc4</version>
</dependency>
<dependency>
<groupId>commons-dbcp</groupId>
<artifactId>commons-dbcp</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.containers</groupId>
<artifactId>jersey-container-servlet</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.ext</groupId>
<artifactId>jersey-spring3</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.media</groupId>
<artifactId>jersey-media-moxy</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
<dependency>
<groupId>org.glassfish.jersey.test-framework.providers</groupId>
<artifactId>jersey-test-framework-provider-inmemory</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.assertj</groupId>
<artifactId>assertj-core</artifactId>
<version>1.5.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.skyscreamer</groupId>
<artifactId>jsonassert</artifactId>
<version>1.2.2</version>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
<repositories>
<repository>
<id>spring-releases</id>
<name>Spring Releases</name>
<url>http://repo.spring.io/libs-release</url>
</repository>
<repository>
<id>org.jboss.repository.releases</id>
<name>JBoss Maven Release Repository</name>
<url>https://repository.jboss.org/nexus/content/repositories/releases</url>
</repository>
</repositories>
<pluginRepositories>
<pluginRepository>
<id>spring-releases</id>
<name>Spring Releases</name>
<url>http://repo.spring.io/libs-release</url>
</pluginRepository>
</pluginRepositories>
</project>
我还有一个Jersey配置(用于Jersey配置):
以及数据库的基本数据源:
package com.example.config;
import java.net.URI;
import java.net.URISyntaxException;
import javax.sql.DataSource;
import org.apache.commons.dbcp.BasicDataSource;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class SimpleDbConfig {
@Bean
public DataSource dataSource() throws URISyntaxException {
URI dbUri;
try {
String username = "username";
String password = "password";
String url = "jdbc:postgresql://localhost/dbname";
String dbProperty = System.getProperty("database.url");
if(dbProperty != null) {
dbUri = new URI(dbProperty);
username = dbUri.getUserInfo().split(":")[0];
password = dbUri.getUserInfo().split(":")[1];
url = "jdbc:postgresql://" + dbUri.getHost() + dbUri.getPath();
}
BasicDataSource basicDataSource = new BasicDataSource();
basicDataSource.setUrl(url);
basicDataSource.setUsername(username);
basicDataSource.setPassword(password);
return basicDataSource;
} catch (URISyntaxException e) {
//Deal with errors here.
throw e;
}
}
}
最后,我有我的REST资源:
package com.example.resource;
import javax.ws.rs.Consumes;
import javax.ws.rs.GET;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import com.example.service.ExampleService;
import com.example.transport.ExampleEntity;
@Path("/entities")
@Component
public class ExampleResource {
private ExampleService exampleService;
@Autowired
public ExampleResource(ExampleService exampleService) {
this.exampleService = exampleService;
}
@POST
@Path("/new")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public ExampleEntity createNewEntity(ExampleEntity entity) {
return exampleService.createNewEntity(entity);
}
@GET
@Path("/{id}")
@Produces(MediaType.APPLICATION_JSON)
public ExampleEntity getExampleDetails(@PathParam("id") Long id) {
return exampleService.findEntity(id);
}
}
服务类别:
package com.example.service;
import javax.transaction.Transactional;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.example.repositories.ExampleRepository;
import com.example.transport.ExampleEntity;
@Service
public class ExampleService {
private ExampleRepository repo;
@Autowired
public ExampleService(ExampleRepository repo) {
this.repo = repo;
}
@Transactional
public ExampleEntity createNewEntity(ExampleEntity entity) {
ExampleEntity savedEntity = repo.save(entity);
return savedEntity;
}
@Transactional
public ExampleEntity findEntity(Long id) {
return repo.findOne(id);
}
}
实体:
package com.example.transport;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
@Entity
public class ExampleEntity {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private long id;
private String name;
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
最后是回购接口:
package com.example.repositories;
import org.springframework.data.repository.CrudRepository;
import com.example.transport.ExampleEntity;
public interface MeetingRepository extends CrudRepository<ExampleEntity, Long> {
}
您似乎还没有创建表
exampleEntity
。我可以看到您正在使用postgreSQL,因此请使用heroku config | grep Herokuu postgreSQL
检查您是否在heroku中启用了一个(以及它的名称和url)。如果没有,请在heroku帐户管理器中启用它,这是一个免费插件。
使用heroku pg:promote heroku\u POSTGRESQL\u
将其升级为主版本。您将获得如下输出:
Promoting HEROKU_POSTGRESQL_BROWN_URL (DATABASE_URL) to DATABASE_URL... done
现在,您应该使用DATABASE_URL环境变量连接到数据库。Heroku会为你提供的。(也请张贴您的文件)
使用url连接到您的数据库,并添加所需的表,其中包含一个处理ExampleEntity
类的模式
旁注
连接到heroku db:
如果我理解错了,您在本地主机上运行应用程序,但尚未将其部署到heroku,那么只需将表添加到本地db实例。我使用的自动表生成示例 除了总是存在的
数据库URL
,Heroku在运行时创建了3个环境变量。它们是:
JDBC_DATABASE_URL
JDBC_DATABASE_USERNAME
JDBC_DATABASE_PASSWORD
您可能知道,如果Spring Boot在应用程序.properties
文件中找到Spring.datasource.*
属性,它将自动配置数据库。下面是我的application.properties的一个示例
spring.datasource.url=${JDBC_DATABASE_URL}
spring.datasource.username=${JDBC_DATABASE_USERNAME}
spring.datasource.password=${JDBC_DATABASE_PASSWORD}
spring.jpa.show-sql=false
spring.jpa.generate-ddl=true
spring.jpa.hibernate.ddl-auto=update
Hibernate/Postgres依赖项
在我的例子中,我使用Hibernate(捆绑在PostgreSQL的spring boot starter jpa
中,因此我需要在我的build.gradle
中使用正确的依赖项:
dependencies {
compile("org.springframework.boot:spring-boot-starter-data-jpa")
compile('org.postgresql:postgresql:9.4.1212')
}
谢谢你的回答。我启用了Postgresql数据库(上面的grep结果类似于:HEROKU_Postgresql_WHITE_URL:postgres:/:@ec2-50-17-207-54.compute-1.amazonaws.com:5432/dfjhng7120830i)。我还使用数据库URL连接到它(通过Procfile传递它并在SimpleDbConfig类中使用它。但是我不知道如何创建我想要使用的表。我以为它们是自动创建的。我如何连接到数据库来创建它们?我可以通过浏览器来做吗?Hibernate可以自动删除并重新创建表,如果你想要的话-只需谷歌
Hibernate即可自动创建
(您实际上不希望在生产环境中执行此操作)。据我所知,您无法使用简单的浏览器连接到数据库,您需要使用数据库客户端上的浏览器。对于postgreSQL,默认的浏览器名为pgadmin,应该随posgresql安装一起提供(…\postgreSQL\9.2\bin
)。我推荐的方法是(在您的计算机上)设置本地数据库,创建必要的表(=关系),然后在本地运行您的应用程序。检查您是否使用带引号的标识符exampleEntity
与“exampleEntity”
是不同的表:
JDBC_DATABASE_URL
JDBC_DATABASE_USERNAME
JDBC_DATABASE_PASSWORD
spring.datasource.url=${JDBC_DATABASE_URL}
spring.datasource.username=${JDBC_DATABASE_USERNAME}
spring.datasource.password=${JDBC_DATABASE_PASSWORD}
spring.jpa.show-sql=false
spring.jpa.generate-ddl=true
spring.jpa.hibernate.ddl-auto=update
dependencies {
compile("org.springframework.boot:spring-boot-starter-data-jpa")
compile('org.postgresql:postgresql:9.4.1212')
}