Исполняемый военный файл, который запускает причал без maven
Я пытаюсь создать "исполняемый" файл войны (java -jar myWarFile.war
), который запустит веб-сервер Jetty, на котором размещается webapp, содержащийся в файле WAR, который я выполнил.
Я нашел страницу, в которой описано, как сделать то, что я ищу:
Однако, следуя этому совету вместе с тем, как я думаю, что я должен сделать исполняемый банку (войну), не работает.
У меня есть задача Ant, создающая WAR файл с манифестом, который выглядит так:
Manifest-Version: 1.0
Ant-Version: Apache Ant 1.7.1
Created-By: 1.5.0_18-b02 (Sun Microsystems Inc.)
Main-Class: Start
Содержимое WAR файла выглядит следующим образом:
> Start.class
> jsp
> build.jsp
> META-INF
> MANIFEST.MF
> WEB-INF
> lib
> jetty-6.1.22.jar
> jetty-util.6.1.22.jar
Когда я пытаюсь выполнить WAR файл, ошибка:
Exception in thread "main" java.lang.NoClassDefFoundError: org/mortbay/jetty/Handler
Caused by: java.lang.ClassNotFoundException: org.mortbay.jetty.Handler
at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:301)
at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
Could not find the main class: Start. Program will exit.
Здесь, по-видимому, две ошибки: одна, где кажется, что файлы JAR не могут быть найдены, и один, где не может быть найден класс Start
.
Чтобы исправить первый, я поместил файлы Jetty JAR в базу WAR файла и попробовал еще раз - ту же ошибку. Я также попытался добавить атрибут WEB-INF/lib/<specific-JAR-files>
к атрибуту Class-Path
манифеста. Это тоже не сработало.
Есть ли у кого-нибудь представление о том, что я делаю правильно/неправильно, и как я могу запустить этот исполняемый файл WAR?
Ответы
Ответ 1
Ссылка , которую вы имеете в своем вопросе, предоставляет большую часть того, что вам нужно. Однако в дополнение к этому нужно сделать несколько вещей.
Любые файлы классов, которые необходимо запустить Jetty, должны быть расположены в корневом файле войны при его упаковке. Мы можем использовать Ant для этого, прежде чем мы <war>
файл. Файл манифеста войны также должен иметь атрибут Main-Class
для выполнения сервера.
Здесь шаг за шагом:
Создайте класс сервера Jetty:
Это адаптировано из предоставленной вами ссылки.
package com.mycompany.myapp;
import java.io.File;
import java.net.URL;
import java.security.ProtectionDomain;
import org.mortbay.jetty.Server;
import org.mortbay.jetty.webapp.WebAppContext;
public final class EmbeddedJettyServer
{
public static void main(String[] args) throws Exception
{
int port = Integer.parseInt(System.getProperty("port", "8080"));
Server server = new Server(port);
ProtectionDomain domain = EmbeddedJettyServer.class.getProtectionDomain();
URL location = domain.getCodeSource().getLocation();
WebAppContext webapp = new WebAppContext();
webapp.setContextPath("/");
webapp.setDescriptor(location.toExternalForm() + "/WEB-INF/web.xml");
webapp.setServer(server);
webapp.setWar(location.toExternalForm());
// (Optional) Set the directory the war will extract to.
// If not set, java.io.tmpdir will be used, which can cause problems
// if the temp directory gets cleaned periodically.
// Your build scripts should remove this directory between deployments
webapp.setTempDirectory(new File("/path/to/webapp-directory"));
server.setHandler(webapp);
server.start();
server.join();
}
}
Чтобы узнать, что вы можете здесь сконфигурировать, посмотрите Документацию API Jetty.
Постройте войну с помощью Ant:
Это использует промежуточную директорию для распаковки необходимых файлов классов в корень войны, чтобы они были доступны, когда война была выполнена.
<target name="war" description="--> Creates self-executing war">
<property name="staging.dir" location="${basedir}/staging"/>
<property name="webapp.dir" location="${basedir}/src/webapp"/>
<mkdir dir="${staging.dir}"/>
<!-- assumes you have all of your war content (excluding classes and libraries) already structured in a directory called src/webapp -->
<!-- e.g. -->
<!-- src/webapp/index.html -->
<!-- src/webapp/WEB-INF/web.xml -->
<!-- src/webapp/WEB-INF/classes/my.properties -->
<!-- etc ... -->
<copy todir="${staging.dir}">
<fileset dir="${webapp.dir}" includes="**/*"/>
</copy>
<unjar dest="${staging.dir}">
<!-- you'll have to locate these jars or appropriate versions; note that these include JSP support -->
<!-- you might find some of them in the downloaded Jetty .tgz -->
<fileset dir="path/to/jetty/jars">
<include name="ant-1.6.5.jar"/>
<include name="core-3.1.1.jar"/>
<include name="jetty-6.1.24.jar"/>
<include name="jsp-2.1-glassfish-2.1.v20091210.jar"/><!-- your JSP implementation may vary -->
<include name="jsp-api-2.1-glassfish-2.1.v20091210.jar"/><!-- your JSP implementation may vary -->
<include name="servlet-api-2.5-20081211.jar"/><!-- your Servlet API implementation may vary -->
</fileset>
<patternset><!-- to exclude some of the stuff we don't really need -->
<exclude name="META-INF/**/*"/>
<exclude name="images/**/*"/>
<exclude name=".options"/>
<exclude name="about.html"/>
<exclude name="jdtCompilerAdapter.jar"/>
<exclude name="plugin*"/>
</patternset>
</unjar>
<!-- copy in the class file built from the above EmbeddedJettyServer.java -->
<copy todir="${staging.dir}">
<fileset dir="path/to/classes/dir" includes="com/mycompany/myapp/EmbeddedJettyServer.class"/>
</copy>
<war destfile="myapp.war" webxml="${webapp.dir}/WEB-INF/web.xml">
<fileset dir="${staging.dir}" includes="**/*"/>
<classes dir="path/to/classes/dir"/><!-- your application classes -->
<lib dir="path/to/lib/dir"/><!-- application dependency jars -->
<manifest>
<!-- add the Main-Class attribute that will execute our server class -->
<attribute name="Main-Class" value="com.mycompany.myapp.EmbeddedJettyServer"/>
</manifest>
</war>
<delete dir="${staging.dir}"/>
</target>
Выполнить войну:
Если все настроено правильно выше, вы должны иметь возможность:
java -jar myapp.war
// or if you want to configure the port (since we are using the System property in the code)
java -Dport=8443 -jar myapp.war
Ответ 2
Это адаптация для ответа Maven @RobHruska. Он просто копирует файлы основного класса и объединяет файлы Jetty JAR в WAR файл, ничего нового, просто для того, чтобы упростить вашу жизнь, если вы новичок, например, с Maven:
<plugin>
<artifactId>maven-antrun-plugin</artifactId>
<executions>
<execution>
<id>move-main-class</id>
<phase>compile</phase>
<configuration>
<tasks>
<copy todir="${project.build.directory}/${project.build.finalName}">
<fileset dir="${project.build.directory}/${project.build.finalName}/WEB-INF/classes/">
<include name="main/*.class" />
</fileset>
</copy>
<unjar dest="${project.build.directory}/${project.build.finalName}">
<!-- you'll have to locate these jars or appropriate versions; note that these include JSP support -->
<!-- you might find some of them in the downloaded Jetty .tgz -->
<fileset dir="${project.build.directory}/${project.build.finalName}/WEB-INF/lib/">
<include name="ant-1.6.5.jar"/>
<!--<include name="core-3.1.1.jar"/>-->
<include name="jetty*"/>
<include name="servlet-api*"/>
</fileset>
<patternset><!-- to exclude some of the stuff we don't really need -->
<exclude name="META-INF/**/*"/>
<exclude name="images/**/*"/>
<exclude name=".options"/>
<exclude name="about.html"/>
<exclude name="jdtCompilerAdapter.jar"/>
<exclude name="plugin*"/>
</patternset>
</unjar>
</tasks>
</configuration>
<goals>
<goal>run</goal>
</goals>
</execution>
</executions>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>2.2</version>
<configuration>
<archiveClasses>true</archiveClasses>
<archive>
<manifest>
<mainClass>main.Main</mainClass>
</manifest>
</archive>
</configuration>
</plugin>
Ответ 3
Мы поняли это, используя плагин-приставку-maven-плагин.
Всякий раз, когда вы запускаете пакет mvn, он создает еще одну войну, которая может использоваться с java -jar whatpackage-runnable.war
<plugin>
<groupId>org.simplericity.jettyconsole</groupId>
<artifactId>jetty-console-maven-plugin</artifactId>
<version>1.45</version>
<executions>
<execution>
<goals>
<goal>createconsole</goal>
</goals>
</execution>
</executions>
<configuration>
<additionalDependencies>
<additionalDependency>
<artifactId>jetty-console-requestlog-plugin</artifactId>
</additionalDependency>
<additionalDependency>
<artifactId>jetty-console-gzip-plugin</artifactId>
</additionalDependency>
<additionalDependency>
<artifactId>jetty-console-ajp-plugin</artifactId>
</additionalDependency>
<additionalDependency>
<artifactId>jetty-console-startstop-plugin</artifactId>
</additionalDependency>
</additionalDependencies>
</configuration>
</plugin>
Он также генерирует скрипты init.d и все для вас!
Ответ 4
Хадсон решает эту проблему, используя контейнер сервлетов Winstone, который напрямую поддерживает этот вариант использования. http://winstone.sourceforge.net/#embedding
Возможно, это сработает для вас?
Ответ 5
Несмотря на то, что это старая альтернатива Jetty 8, просто включите банки Jetty в качестве зависимостей в вашем pom и добавьте следующее в ваш pom (по сравнению с ant script, который распаковывает войну и переупаковывает ее ):
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>1.4</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<createDependencyReducedPom>true</createDependencyReducedPom>
<transformers>
<transformer
implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>JettyStandaloneMain</mainClass>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
<!-- The main class needs to be in the root of the war in order to be
runnable -->
<plugin>
<artifactId>maven-antrun-plugin</artifactId>
<executions>
<execution>
<id>move-main-class</id>
<phase>compile</phase>
<configuration>
<tasks>
<move todir="${project.build.directory}/${project.build.finalName}">
<fileset dir="${project.build.directory}/classes/">
<include name="JettyStandaloneMain.class" />
</fileset>
</move>
</tasks>
</configuration>
<goals>
<goal>run</goal>
</goals>
</execution>
</executions>
</plugin>
Ответ 6
Я считаю, что "без maven" вам нужен банку, которую вы можете запустить самостоятельно, а не "mvn jetty: run" - не то, что вы вообще не хотите использовать maven.
Мне потребовалось много времени, чтобы понять это, потому что я нашел много вариантов - ни одна из них не была простой. В конце концов я нашел этот плагин maven от простоты. Он прекрасно работает.
Ответ 7
Это мой пример ANT extract. Идея состоит в том, чтобы распаковывать зависимости Jetty, а затем включать их локально, как обычный JAR файл:
<!-- Hack: Java doesn't support jars within jars/wars -->
<unjar src="${lib.dir}/container/jetty.jar" dest="${build.dir}/unjar"/>
<unjar src="${lib.dir}/container/jetty-util.jar" dest="${build.dir}/unjar"/>
<unjar src="${lib.dir}/container/servlet-api.jar" dest="${build.dir}/unjar"/>
<unjar src="${lib.dir}/container/jsp-api.jar" dest="${build.dir}/unjar"/>
<!-- Build war file as normal, just including the compiled and unjar'ed files -->
<war destfile="${war.file}" webxml="${config.dir}/web.xml">
<fileset dir="${build.dir}/classes"/>
<fileset dir="${build.dir}/unjar"/>
<fileset dir="${resources.dir}" excludes="*.swp"/>
<lib dir="${lib.dir}/runtime"/>
<manifest>
<attribute name="Main-Class" value="Start"/>
</manifest>
</war>
Примечание:
Директива WEB-INF/lib предназначена для зависимостей веб-приложений. В этом случае мы собираем файл WAR так, чтобы он работал, как обычный файл JAR Jetty при запуске
Ответ 8
- Ввод .jars внутри корня файла .war ничего не делает
- Ввод .jars внутри
WEB-INF/lib
не помогает JVM найти файлы Jetty даже при запуске вашего .war. "Слишком поздно", чтобы положить их туда.
- Ввод .jars в манифесте Class-Path работает только для внешних .jar файлов, а не для тех, которые содержатся в .jar
Итак, что делать?
- Используйте build script, чтобы просто объединить все файлы .jar, которые вам нужны, в .war файл. Это требует немного дополнительной работы. Это также немного уродливо в том, что скомпилированный код является частью поддерживаемых файлов в .war
- Добавить зависимые .jars в JVM classpath с помощью "java -cp jetty.jar:......" Работает, несмотря на то, что это поражает цель одного автономного .war
Ответ 9
Я уже делал подобное, но вы запускаете приложение как "java -jar xxx.war"? Думаю, у вас всего 2 баночки, и этого не будет. Также попробуйте использовать Jetty 7.0.0M1 (которая является последней версией). Когда я добавил jetty-server и jetty-webapp как две зависимости (они из org.eclipse.jetty), я получаю следующую банку в каталоге lib. FYI org.mortbay.jetty.Handler был в причальном сервере *.jar.
- Причал-продолжение 7.0.0.M1.jar
- Причал-HTTP-7.0.0.M1.jar
- Причал-ю-7.0.0.M1.jar
- Причал-безопасности 7.0.0.M1.jar
- Причал-сервер 7.0.0.M1.jar
- Причал-сервлет-7.0.0.M1.jar
- jetty-util-7.0.0.M1.jar
- Причал-WebApp-7.0.0.M1.jar
- Причал-XML-7.0.0.M1.jar