gpt4 book ai didi

java - 构建一次并部署到多个环境

转载 作者:行者123 更新时间:2023-11-30 08:54:05 25 4
gpt4 key购买 nike

我们有包含 50 -60 个 maven 模块的遗留系统,所有这些模块都使用 maven 资源插件来过滤属性(在构建时替换属性文件中的标记)。当涉及到为不同的环境构建时,这是非常痛苦的,因为每次我们需要为不同的环境部署时,我们都必须构建应用程序。

我们有一个新的要求,即一次构建应用程序并为多个环境部署。最好的解决方案是什么?我在考虑外部化过滤器属性,但最大的问题是替换下面应用程序的现有属性文件(请参阅 application.properties)文件的标记。我想保持现有的属性文件不变,并从外部配置文件中选择值。

如有任何帮助,我们将不胜感激。

例如注入(inject)maven的Filter.properties。

generic.sharepoint.host=xxxxx
generic.deploy.apps.host=xxxxx
generic.deploy.apps.url=xxxx
generic.deploy.trusted.host=xxxx
generic.deploy.trusted.url=xxxx
generic.deploy.orderentry=xxxxx

application.properties

generic.sharepoint.host=${generic.sharepoint.host}
generic.deploy.apps.host=${generic.deploy.apps.host}
generic.deploy.apps.url=${generic.deploy.apps.url}
generic.deploy.trusted.host=${generic.deploy.trusted.host}
generic.deploy.trusted.url=${generic.deploy.trusted.url}
generic.deploy.orderentry=${generic.deploy.orderentry}

最佳答案

根据我的经验,我还必须构建一次基于 Spring 的 Web 应用程序并部署到许多不同的环境中,因此我试图向您展示适合我的解决方案。

我找不到将这个任务交给 Maven 的方法,所以我决定将配置外部化,并发现利用 Spring 我可以设法实现只执行两个步骤:

  1. 通过读取包含环境名称(例如 DEV、TEST、PRODUCTION)的文件,让应用程序“了解”它所部署到的环境
  2. 根据找到的环境值,设置一个系统属性,然后加载对应文件夹下的配置文件。

有了 Tomcat,我们在 shared/classes 下设置配置文件夹结构,如下所示:

config structure

然后我们将包含环境描述的application-config.properties 文件放入application-config 文件夹:

application-config.env=DEV

以及每个子文件夹下的相同配置文件正确配置为相关环境所需。之后,我们必须向 Spring 应用程序上下文添加另一个带有一些 bean 定义的 application-config-context.xml,以便与 org.springframework.beans.factory.config.PropertiesFactoryBean 进行交互:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd">

<bean id="environmentDetector" class="com.mycompany.myapp.spring.util.PropertiesBasedEnvironmentDetector" init-method="init" >
<property name="properties" ref="environmentDefinitionProperties" />
<property name="environmentDefinitionPropertyName" value="application-config.env"/>
</bean>

<bean id="environmentDefinitionProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean">
<property name="ignoreResourceNotFound" value="true" />
<property name="locations">
<list>
<value>classpath:application-config/application-config.properties</value>
</list>
</property>
</bean>

<bean
class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer" >
<property name="systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />
<property name="ignoreResourceNotFound" value="false" />
<property name="ignoreUnresolvablePlaceholders" value="true" />
<property name="properties" ref="applicationExternalProperties" />
</bean>

<bean id="applicationExternalProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean" >
<property name="location" value="${application-config.prefix}application-config/${application-config.env}/application.properties" />
</bean>

</beans>

我必须编写一个类来覆盖接口(interface) org.springframework.beans.factory.config.BeanFactoryPostProcessor 的 Spring postProcessBeanFactory 方法,如下所示:

package com.mycompany.myapp.doc.spring.util;

import java.util.Properties;

import javax.annotation.PostConstruct;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.core.PriorityOrdered;

public class PropertiesBasedEnvironmentDetector implements BeanFactoryPostProcessor, PriorityOrdered{
private static final Log log = LogFactory.getLog(PropertiesBasedEnvironmentDetector.class);

private Properties properties;
private String environmentDefinitionPropertyName ="env";
private String defaultEnvironment="DEV";
private String environmentSystemPropertyName;
private int order = PriorityOrdered.HIGHEST_PRECEDENCE;

private String prefixSystemPropertyName = "application-config.prefix";
private String prefixDefault = "classpath:";
private String prefix;

public void setProperties(Properties properties) {
this.properties = properties;
}

@PostConstruct
public void init()
{
if (environmentSystemPropertyName == null) {
environmentSystemPropertyName = environmentDefinitionPropertyName;
}
String activeEnvironment = properties.getProperty(environmentDefinitionPropertyName, defaultEnvironment);
prefix = properties.getProperty(prefixSystemPropertyName);
if (prefix == null) {
prefix = prefixDefault;
properties.put(prefixSystemPropertyName, prefix);
}
System.setProperty(environmentSystemPropertyName , activeEnvironment);
System.setProperty(prefixSystemPropertyName , prefix);

log.warn("Initializing Environment: "+activeEnvironment);

}

public String getEnvironmentDefinitionPropertyName() {
return environmentDefinitionPropertyName;
}

public void setEnvironmentDefinitionPropertyName(
String environmentDefinitionPropertyName) {
this.environmentDefinitionPropertyName = environmentDefinitionPropertyName;
}

public String getDefaultEnvironment() {
return defaultEnvironment;
}

public void setDefaultEnvironment(String defaultEnvironment) {
this.defaultEnvironment = defaultEnvironment;
}

public String getEnvironmentSystemPropertyName() {
return environmentSystemPropertyName;
}

public void setEnvironmentSystemPropertyName(String environmentSystemPropertyName)
{
this.environmentSystemPropertyName = environmentSystemPropertyName;
}

@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory)
throws BeansException
{
String activeEnvironment = System.getProperty(environmentSystemPropertyName);
log.warn("PostProcessing ApplicationContext for Environment: "+activeEnvironment+" from "+prefix);
}

@Override
public int getOrder() {
return order;
}

public void setOrder(int order) {
this.order = order;
}

}

当应用程序启动时,Spring 加载所有上下文并调用我们的自定义类。在 init() 方法中,首先通过注入(inject)的 properties 属性加载 environmentDefinitionPropertyName,然后将其设置为系统属性,并以 environmentDefinitionPropertyName 在 bean 定义中设置的值。

之后,PropertyPlaceholderConfigurer 可以加载属性文件位置,因为它解析了:

<property name="location" value="${application-config.prefix}application-config/${application-config.env}/application.properties" />

进入

<property name="location" value="classpath:application-config/DEV/application.properties" />

这种方法具有以下主要优点:

  • 您可以轻松添加更多环境,只需在原始配置文件夹下添加另一个文件夹
  • 原始应用程序包在所有环境中都是相同的
  • 属性文件具有“正常”key=value 格式,就像您的 application.properties 文件一样。

希望这对您有所帮助,也可能对其他人有所帮助。

关于java - 构建一次并部署到多个环境,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29552425/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com