Spring Boot YML 和独立 Tomcat 8 服务器

Posted

技术标签:

【中文标题】Spring Boot YML 和独立 Tomcat 8 服务器【英文标题】:Spring Boot YML and StandAlone Tomcat 8 Server 【发布时间】:2015-11-26 10:46:10 【问题描述】:

我有以下目录结构/配置文件:

src/main/resource/config: 
application.yml 
application-dev.yml 
application-sit.yml

注意根据“引导配置”https://spring.io/blog/2015/01/13/configuring-it-all-out-or-12-factor-app-style-configuration-with-spring:

Spring Boot 会读取 默认情况下在 src/main/resources/application.properties 中的属性。如果 配置文件处于活动状态,它也会自动读取 基于配置文件名称的配置文件,例如 src/main/resources/application-foo.properties 其中 foo 是当前 轮廓。如果 Snake YML 库在类路径上,那么它将 还会自动加载 YML 文件。

如果我将--spring.profiles.active=dev 设置为 eclipse 运行配置中的程序 arg 并将其用作我的主要方法任何事情都按预期工作:

  public static void main(String[] args) 
        SpringApplication app = new SpringApplication(Application.class);

        SimpleCommandLinePropertySource source = new SimpleCommandLinePropertySource(args);

        // Check if the selected profile has been set as argument.
        // if not the development profile will be added
        addDefaultProfile(app, source);

        app.run(args);
    

    /**
     * Set a default profile if it has not been set
     */
    private static void addDefaultProfile(SpringApplication app, SimpleCommandLinePropertySource source) 
        if (!source.containsProperty("spring.profiles.active")) 
            app.setAdditionalProfiles(Constants.SPRING_PROFILE_DEVELOPMENT);
        
    

(请注意上面的主要方法引用来自我代码中使用的以下类:https://github.com/jarias/generator-jhipster-ember/blob/master/app/templates/src/main/java/package/_Application.java)

对于 spring.profile.active=dev,一切都按预期工作。这意味着两者: application.yml(默认加载)和 application-dev.yml(活动配置文件)属性文件已加载并排除 application- sat.yml 因为 sat 不是一个活跃的配置文件。

这个嵌入式容器非常适合开发测试。但是,我想通过生成战争并将其部署到独立的 Tomcat8 服务器来将其发布到生产中。

为此,我创建了一个 WebApplicationInitializer 实现,Tomcat8 服务器需要它来自动检测、引导和启动独立服务器上的 Spring 应用程序。

@Configuration
public class WebAppInit implements WebApplicationInitializer 


    @Override
    public void onStartup(ServletContext servletContext) throws ServletException 
        ApplicationContext context = new AnnotationConfigApplicationContext(Application.class);
        

部署战争后,我收到以下错误,我尝试启动独立服务器并收到以下错误:

引起:org.springframework.beans.factory.enter code hereBeanCreationException:无法自动装配字段:私有 java.lang.String com.titlefeed.config.db.DbConfigJPA.databaseUrl; 嵌套异常是 java.lang.IllegalArgumentException: 不能 在字符串值中解析占位符 spring.data.postgres.uri' "$spring.data.postgres.uri"

这意味着 Tomcat 服务器/Spring 没有加载 application-dev.yml,因为它包含以下属性:spring.data.postgres.uri

所以我尝试了以下两种解决方案

    在 tomcat/bin/catalina.sh 的 JAVA_OPTS 中添加了 -Dspring.profiles.active=devspring.profiles.active=dev 添加到tomcat/conf/catalina.properties

他们都没有工作。如何让独立的 tomcat 服务器加载与 spring.profiles.active 属性关联的 yml 文件。

它适用于从 eclipse 启动的嵌入式 springboot 服务器,但不适用于独立服务器?

EDIT1:M. Deinum - 在下面实施了您建议的解决方案,但仍然出现以下错误:

Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder 'spring.data.postgres.uri' in string value "$spring.data.postgres.uri

似乎 -Dspring.profiles.active=dev 没有设置。

@Configuration
public class WebAppInit extends SpringBootServletInitializer 

 @Override

    protected WebApplicationContext createRootApplicationContext(
            ServletContext servletContext) 
           log.info("Properly INITALIZE spring CONTEXT");
           ApplicationContext context = new AnnotationConfigApplicationContext(Application.class);
           servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, context);
           return super.createRootApplicationContext(servletContext);
    


EDIT 2 ACV: - 在启动脚本中添加“--spring.profiles.active=dev”作为 JAVA_OPTS 变量的一部分:tomcat/bin/catalina.sh 不是一个可行的选项

例如:

 JAVA_OPTS="$JAVA_OPTS --spring.profiles.active=dev ...etc

给出以下错误:

无法识别的选项:--spring.profiles.active=dev 错误:无法 创建 Java 虚拟机。”

编辑 3: 修改了 application.yml 以包含以下属性

spring:
  profiles:
    active: dev

重新部署战争。转到分解的 tomcat 目录位置以确保该属性存在 webapps/feedserver/WEB-INF/classes/config/application.yml

问题仍然存在。

编辑 4: 在 tomcat 爆炸的 webdir 下添加 application.properties:webapps/feedserver/WEB-INF/classes/application.properties:

spring.profiles.active=dev
spring.data.postgres.uri=jdbc:postgresql://localhost:5432/feedserver

重启tomcat,问题依旧。

它似乎没有选择 application.propertiesapplication.yml

EDIT 5 使用推荐的方式为外部容器启动 Spring Boot 服务器:

@Configuration
public class WebAppInit extends SpringBootServletInitializer 

 @Override
 protected SpringApplicationBuilder configure(SpringApplicationBuilder application) 
     return application.sources(Application.class);
 


编辑 6:

我在启动命令 args 中添加了 -Dspring.profiles.active=dev

/Library/Java/JavaVirtualMachines/jdk1.8.0_25.jdk/Contents/Home/bin/java -Djava.util.logging.config.file=/Users/shivamsinha/Desktop/Programming/tomcat/conf/logging.properties -Djava.util.logging.manager=org.apache.juli.ClassLoaderLogManager -Dlog4j.rootLevel=ERROR -Dlog4j.rootAppender=console -DENV=dev -Dlog4j.configuration=/WEB-INF/classes/properties/log4j.properties -DTOMCAT_DIR=WEB-INF/classes/ -Djava.endorsed.dirs=/Users/shivamsinha/Desktop/Programming/tomcat/endorsed -classpath /Users/shivamsinha/Desktop/Programming/tomcat/bin/bootstrap.jar:/Users/shivamsinha/Desktop/Programming/tomcat/bin/tomcat-juli.jar -Dcatalina.base=/Users/shivamsinha/Desktop/Programming/tomcat -Dcatalina.home=/Users/shivamsinha/Desktop/Programming/tomcat -Djava.io.tmpdir=/Users/shivamsinha/Desktop/Programming/tomcat/temp org.apache.catalina.startup.Bootstrap -Dspring.profiles.active=dev start

但是我仍然在日志中得到以下异常:

Caused by: org.springframework.beans.factory.BeanCreationException: Could not autowire field: private java.lang.String com.titlefeed.config.db.DbConfigJPA.databaseUrl; nested exception is java.lang.IllegalArgumentException: Could not resolve placeholder 'spring.data.postgres.uri' in string value "$spring.data.postgres.uri"
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:561)
    at org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:88)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:331)
    ... 68 more
Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder 'spring.data.postgres.uri' in string value "$spring.data.postgres.uri"
    at org.springframework.util.PropertyPlaceholderHelper.parseStringValue(PropertyPlaceholderHelper.java:174)
    at org.springframework.util.PropertyPlaceholderHelper.replacePlaceholders(PropertyPlaceholderHelper.java:126)
    at org.springframework.core.env.AbstractPropertyResolver.doResolvePlaceholders(AbstractPropertyResolver.java:204)
    at org.springframework.core.env.AbstractPropertyResolver.resolveRequiredPlaceholders(AbstractPropertyResolver.java:178)
    at org.springframework.context.support.PropertySourcesPlaceholderConfigurer$2.resolveStringValue(PropertySourcesPlaceholderConfigurer.java:175)
    at org.springframework.beans.factory.support.AbstractBeanFactory.resolveEmbeddedValue(AbstractBeanFactory.java:801)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:955)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:942)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:533)
    ... 70 more

02-Sep-2015 03:15:40.472 SEVERE [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployWAR Error deploying web application archive /Users/shivamsinha/Desktop/Programming/tomcat/webapps/feedserver-1.0.0.war
 java.lang.IllegalStateException: ContainerBase.addChild: start: org.apache.catalina.LifecycleException: Failed to start component [StandardEngine[Catalina].StandardHost[localhost].StandardContext[/feedserver-1.0.0]]
    at org.apache.catalina.core.ContainerBase.addChildInternal(ContainerBase.java:728)
    at org.apache.catalina.core.ContainerBase.addChild(ContainerBase.java:701)
    at org.apache.catalina.core.StandardHost.addChild(StandardHost.java:714)
    at org.apache.catalina.startup.HostConfig.deployWAR(HostConfig.java:917)
    at org.apache.catalina.startup.HostConfig$DeployWar.run(HostConfig.java:1701)
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)

【问题讨论】:

docs.spring.io/spring-boot/docs/current/reference/html/… 尝试设置:--spring.profiles.active=dev 因为你做错了,使用特殊SpringBootServletInitializer的子类。如弹簧靴参考指南中所述。 我认为我们有同样的问题。见***.com/questions/32326965/…不过目前没有解决方案吧? @M.Deinum 我已经实施了你的建议。但是我认为问题在于 spring.profiles.active 没有被设置。 @ACV 关于使用 -spring.profiles.active=dev 的问题,请参阅我在问题中的新编辑。我应该在哪里设置这个? 【参考方案1】:

信用:@M。迪南

将 spring 配置文件参数传递到 Tomcat 8 有两个选项。

1.设置为环境变量

Tomcat 允许您在启动过程中调用的CATALINA_HOME/setenv.shCATALINA_BASE/setenv.sh 中设置环境配置。

setenv.sh:
export SPRING_PROFILES_ACTIVE=dev

您可能还想创建一个 src/main/resources/banner.txt 并在其中包含此行:

active profiles     :: $spring.profiles.active

它在您的 IDE 中不起作用(它从您的 jar/war 的 MANIFEST.MF 文件中读取,如果您正常编译则不会拥有该文件),但它在您关心的环境中非常方便——除了您当地的环境之外的一切!

2.将其添加到执行类之前的启动脚本/命令之前

我修改CATALINA_HOME/catalina.sh添加了一个声明变量:

SPRING_PROFILE="dev"

并且在所有相关的执行中都将其添加到脚本中:

  eval "\"$_RUNJAVA\"" "\"$LOGGING_CONFIG\"" $LOGGING_MANAGER $JAVA_OPTS $CATALINA_OPTS \
      -Djava.endorsed.dirs="\"$JAVA_ENDORSED_DIRS\"" -classpath "\"$CLASSPATH\"" \
      -Dcatalina.base="\"$CATALINA_BASE\"" \
      -Dcatalina.home="\"$CATALINA_HOME\"" \
      -Djava.io.tmpdir="\"$CATALINA_TMPDIR\"" \
      -Dspring.profiles.active="\"$SPRING_PROFILE\"" \
      org.apache.catalina.startup.Bootstrap "$@" start \
      >> "$CATALINA_OUT" 2>&1 "&"

显然这不是推荐的方法。但它有效!如果您确实有确切的可复制步骤,推荐的方法随时发布答案,如果可行,我会接受。

【讨论】:

在 UNIX 变体上使用 SPRING_PROFILES_ACTIVE 也可以。 将 setenv.sh 添加到 bin/ export SPRING_PROFILES_ACTIVE=dev 然后重启 tomcat 没有任何反应。【参考方案2】:

我在寻找将 spring 配置文件传递给独立的 tomcat 服务器的解决方案时遇到了这个问题。如果有人遇到同样的问题,对我有用的是将属性添加到 tomcat catalina.properties - 一行:spring.profiles.active=dev

【讨论】:

【参考方案3】:

尝试从配置路径中删除“config”子文件夹。不是所有的配置文件都在src/main/resources 中吗?

【讨论】:

迪米特里。默认情况下,我们也会查找config,请检查the documentation 对,对不起。我查看了最初的博客文章文档。应该将建议作为评论发布,而不是作为答案。

以上是关于Spring Boot YML 和独立 Tomcat 8 服务器的主要内容,如果未能解决你的问题,请参考以下文章

Spring boot jsp 和 application.yml

Spring Boot 中application.yml与bootstrap.yml的区别

spring boot修改yml 端口号不生效解决方案

Spring Boot 引入自定义yml

Spring Boot总结

Java面试之Spring Boot/Spring Cloud