Spring Boot: Как добавить еще один WAR-файл во встроенный tomcat?
Встроенный tomcat в Spring Boot очень удобен как для разработки, так и для развертывания.
Но что, если нужно добавить еще один (сторонний) WAR-файл (например, GeoServer)?
Возможно, следующая процедура является нормальной:
- Установите обычный сервер Tomcat.
- Создайте приложение Spring Boot в виде файла WAR и добавьте его в папку webapps Tomcat.
- Также добавьте другой (сторонний) WAR-файл в папку webapps.
Но было бы неплохо, если бы была возможна следующая конфигурация.
- Создайте загрузочное приложение Spring в виде отдельного Jar-файла, включающего встроенный Tomcat.
- Разверните загрузочное приложение Spring Jar.
- Добавьте еще один (сторонний) WAR-файл в папку, которую распознает встроенный Tomcat.
- Обслуживайте содержимое загрузочного приложения Spring и содержимое другой WAR с помощью встроенного Tomcat.
Как это можно сделать?
ОБНОВИТЬ
Когда весеннее загрузочное приложение сделано из толстого jar(= исполняемый jar), кода в ответе недостаточно. Пересмотрено следующее:
@Bean
public EmbeddedServletContainerFactory servletContainerFactory() {
return new TomcatEmbeddedServletContainerFactory() {
@Override
protected TomcatEmbeddedServletContainer getTomcatEmbeddedServletContainer(
Tomcat tomcat) {
try {
Context context = tomcat.addWebapp("/foo", "/path/to/foo.war");
WebappLoader loader =
new WebappLoader(Thread.currentThread().getContextClassLoader());
context.setLoader(loader);
} catch (ServletException ex) {
throw new IllegalStateException("Failed to add webapp", ex);
}
return super.getTomcatEmbeddedServletContainer(tomcat);
}
};
}
Поскольку файлы jar в толстом jar не могут быть загружены системным загрузчиком классов, необходимо указать явный родительский загрузчик классов. В противном случае дополнительная WAR не сможет загрузить библиотеки библиотек в толстую банку приложения весенней загрузки, которое добавило WAR.
1 ответ
Вы можете добавить военный файл во встроенный Tomcat, используя Tomcat.addWebapp
, Как говорит его Javadoc, это "эквивалент добавления веб-приложения в каталог веб-приложений Tomcat". Чтобы использовать этот API в Spring Boot, вам нужно использовать пользовательский TomcatEmbeddedServletContainerFactory
подкласс:
@Bean
public EmbeddedServletContainerFactory servletContainerFactory() {
return new TomcatEmbeddedServletContainerFactory() {
@Override
protected TomcatEmbeddedServletContainer getTomcatEmbeddedServletContainer(
Tomcat tomcat) {
// Ensure that the webapps directory exists
new File(tomcat.getServer().getCatalinaBase(), "webapps").mkdirs();
try {
Context context = tomcat.addWebapp("/foo", "/path/to/foo.war");
// Allow the webapp to load classes from your fat jar
context.setParentClassLoader(getClass().getClassLoader());
} catch (ServletException ex) {
throw new IllegalStateException("Failed to add webapp", ex);
}
return super.getTomcatEmbeddedServletContainer(tomcat);
}
};
}
Принятый ответ касается Spring Boot 1.x. Упомянутый класс больше не присутствует в Spring Boot 2.x. При использовании версии 2 вам необходимо использовать другую:
@Bean
@ConditionalOnProperty(name = "external.war.file")
public TomcatServletWebServerFactory servletContainerFactory(@Value("${external.war.file}") String path,
@Value("${external.war.context:}") String contextPath) {
return new TomcatServletWebServerFactory() {
@Override
protected TomcatWebServer getTomcatWebServer(Tomcat tomcat) {
new File(tomcat.getServer().getCatalinaBase(), "webapps").mkdirs();
Context context = tomcat.addWebapp(contextPath, path);
context.setParentClassLoader(getClass().getClassLoader());
return super.getTomcatWebServer(tomcat);
}
};
}
Кроме того, Tomcat со встроенной загрузкой Spring по умолчанию не содержит зависимостей для JSP. Если вы используете JSP во внешней войне, вам необходимо включить их.
<dependency>
<groupId>org.apache.tomcat.embed</groupId>
<artifactId>tomcat-embed-jasper</artifactId>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jstl</artifactId>
</dependency>
ОБНОВЛЕНИЕ: я написал более подробное сообщение в блоге о том, как настроить это для Spring Boot 1 и 2.
Потребовалось время, чтобы понять это для Spring Boot 2, поскольку ни один из ответов не помог мне полностью. Наконец, я придумал это (кстати, у меня включен SSL): WarRun.java с зависимостями Gradle ниже, чтобы он работал.
Что это дает:
встроенный tomcat с контекстным путем / на https://localhost:8070/
sample.war по адресу https://localhost:8070/sample
SampleWebApp.war по адресу https://localhost:8070/yo.
import java.io.File;
import java.io.IOException;
import java.io.PrintStream;
import java.util.Properties;
import org.apache.catalina.Context;
import org.apache.catalina.startup.Tomcat;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.WebApplicationType;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.embedded.tomcat.TomcatWebServer;
import org.springframework.boot.web.servlet.server.ServletWebServerFactory;
import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Profile;
import org.springframework.core.io.ClassPathResource;
@ComponentScan({ "com.towianski.controllers" })
@SpringBootApplication
@Profile("server")
public class WarRun extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
return application.sources(WarRun.class).web( WebApplicationType.SERVLET );
}
public static void main(String[] args) {
SpringApplication app = new SpringApplication(WarRun.class);
System.out.println( "Entered WarRun.main");
String loggingFile = "";
String dir = "";
for ( int i = 0; i < args.length; i++ )
{
// logger.info( "** args [" + i + "] =" + args[i] + "=" );
System.out.println( "** args [" + i + "] =" + args[i] + "=" );
if ( args[i].toLowerCase().startsWith( "-dir" ) )
{
dir = args[i].substring( "-dir=".length() );
}
else if ( args[i].toLowerCase().startsWith( "--logging.file" ) )
{
loggingFile = args[i].substring( "--logging.file=".length() );
stdOutFilePropertyChange( loggingFile );
stdErrFilePropertyChange( loggingFile );
}
}
Properties properties = new Properties();
// properties.setProperty( "spring.resources.static-locations",
// "classpath:/home/stan/Downloads" );
properties.setProperty( "server.port", "8070" );
// System.setProperty("server.servlet.context-path", "/prop"); <--- Will set embedded Spring Boot Tomcat context path
properties.setProperty( "spring.security.user.name", "stan" );
properties.setProperty( "spring.security.user.password", "stan" );
System.out.println( "Entered WarRun.main after set properties");
app.setDefaultProperties(properties);
System.out.println( "Entered WarRun.main after call set props. before app.run");
app.run(args);
System.out.println( "Entered WarRun.main after app.run()");
}
@Bean
public ServletWebServerFactory servletContainer() {
return new TomcatServletWebServerFactory() {
protected TomcatWebServer getTomcatWebServer(Tomcat tomcat) {
System.out.println( "tomcat.getServer().getCatalinaBase() =" + tomcat.getServer().getCatalinaBase() + "=" );
new File(tomcat.getServer().getCatalinaBase(), "/webapps").mkdirs();
// try {
// Files.copy( (new File( "/home/stan/Downloads/sample.war" ) ).toPath(), (new File( tomcat.getServer().getCatalinaBase() +"/webapp/sample.war") ).toPath());
// } catch (IOException ex) {
// Logger.getLogger(WarRun.class.getName()).log(Level.SEVERE, null, ex);
// }
try {
System.out.println( "Entered ServletWebServerFactory servletContainer()");
Context context2 = tomcat.addWebapp("/sample", new ClassPathResource("file:/home/stan/Downloads/sample.war").getFile().toString());
Context context3 = tomcat.addWebapp("/yo", new ClassPathResource("file:/home/stan/Downloads/SampleWebApp.war").getFile().toString());
// Context context = tomcat.addWebapp("/what", new ClassPathResource( "file:" + tomcat.getServer().getCatalinaBase() +"/webapps/sample.war" ).getFile().toString() );
context2.setParentClassLoader(getClass().getClassLoader());
context3.setParentClassLoader(getClass().getClassLoader());
// also works but above seems better
// WebappLoader loader2 = new WebappLoader(Thread.currentThread().getContextClassLoader());
// WebappLoader loader3 = new WebappLoader(Thread.currentThread().getContextClassLoader());
// context2.setLoader(loader2);
// context3.setLoader(loader3);
} catch (IOException ex) {
ex.printStackTrace();
}
return super.getTomcatWebServer(tomcat);
}
};
}
}
Gradle:
apply plugin: 'war'
war {
enabled = true
}
. . . .
dependencies {
compile("org.springframework.boot:spring-boot-starter:2.1.6.RELEASE")
compile("org.springframework.boot:spring-boot-starter-web:2.1.6.RELEASE")
compile group: 'org.apache.tomcat.embed', name: 'tomcat-embed-jasper', version: '9.0.21'
compile("org.springframework.boot:spring-boot-starter-security:2.1.6.RELEASE")
compile 'org.apache.httpcomponents:httpclient:4.5.7'
compile group: 'org.codehaus.groovy', name: 'groovy-all', version: '2.5.6'
compile fileTree(dir: 'libs', include: ['*.jar'])
compile 'com.jcraft:jsch:0.1.55'
testCompile group: 'junit', name: 'junit', version: '4.12'
}