Отдельный pushfilter: jetty.server.request not found
Я работаю над автономным фильтром для расширения нашего веб-приложения (BaseX), если оно используется с Jetty. Цель состоит в том, чтобы использовать некоторые дополнительные ресурсы при отправке ответа. Веб-приложение определяет, какие ресурсы должны быть отправлены.
Со встроенным причалом все идет хорошо. Но при развертывании сервлета BaxeX als на jetty и запуске с jettx:run-forked я получаю следующее:
java.lang.NoClassDefFoundError: org/eclipse/jetty/server/Request
at org.basex.http2.BaseXJettyPushFilter.doFilter(BaseXJettyPushFilter.java:48)
at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1637)
at org.eclipse.jetty.servlet.ServletHandler.doHandle(ServletHandler.java:533)
at org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:143)
at org.eclipse.jetty.security.SecurityHandler.handle(SecurityHandler.java:548)
at org.eclipse.jetty.server.handler.HandlerWrapper.handle(HandlerWrapper.java:132)
at org.eclipse.jetty.server.handler.ScopedHandler.nextHandle(ScopedHandler.java:190)
at org.eclipse.jetty.server.session.SessionHandler.doHandle(SessionHandler.java:1595)
...
Это pom.xml фильтра:
<?xml version="1.0"?>
<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>org.basex</groupId>
<artifactId>basex-jetty-push-filter</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>jar</packaging>
<name>basex-jetty-push-filter</name>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<jetty.version>9.4.6.v20170531</jetty.version>
</properties>
<dependencies>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-annotations</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-servlet</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-servlets</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-server</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-alpn-server</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty.http2</groupId>
<artifactId>http2-common</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty.http2</groupId>
<artifactId>http2-server</artifactId>
<version>${jetty.version}</version>
</dependency>
<dependency>
<groupId>org.eclipse.jetty</groupId>
<artifactId>jetty-webapp</artifactId>
<version>${jetty.version}</version>
</dependency>
</dependencies>
<build>
<pluginManagement>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.7.0</version>
<configuration>
<!-- http://maven.apache.org/plugins/maven-compiler-plugin/ -->
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
</plugins>
</pluginManagement>
</build>
</project>
...
и я загружаю его как зависимость здесь:
<parent>
<groupId>org.basex</groupId>
<artifactId>basex-parent</artifactId>
<version>9.0-SNAPSHOT</version>
<relativePath>..</relativePath>
</parent>
<name>BaseX API</name>
<dependencies>
<dependency>
<groupId>org.basex</groupId>
<artifactId>basex</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.basex</groupId>
<artifactId>basex-jetty-push-filter</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>net.xqj</groupId>
<artifactId>basex-xqj</artifactId>
</dependency>
<dependency>
<groupId>org.xmldb</groupId>
<artifactId>xmldb-api</artifactId>
</dependency>
...
Понятно, почему я не могу получить доступ к серверным классам из веб-приложения! Веб-приложение не должно иметь никаких зависимостей от окружающего сервлет-контейнера, потому что оно может измениться.
Вот почему я реализовал фильтр автономно, но каким-то образом он все еще не может получить доступ к серверу. Класс запроса. PushCacheFilter от Jetty также использует класс jetty.request, это должно быть возможно!
Вот фильтр: https://github.com/BodoWissemann/basex-jetty-push-filter
Как я могу решить эту проблему? Я застрял:(
Большое спасибо!
Бодо
1 ответ
Симона Бордет ответила на мой вопрос:
"@BodoWissemann, вы правильно поняли, что классы Servlet Container должны быть скрыты для развернутых веб-приложений, и именно поэтому вы получаете NoClassDefFoundError при развертывании веб-приложения в Jetty, а не при запуске его встроенным.
PushCacheFilter от Jetty работает, потому что у нас есть правило системных классов по умолчанию для него в загрузчике классов веб-приложения, см. https://github.com/eclipse/jetty.project/blob/jetty-9.4.x/jetty-webapp/src/main/java/org/eclipse/jetty/webapp/WebAppContext.java#L128.
Вы должны сделать то же самое для вашего BaseXJettyPushFilter: поместите его в путь к классу сервера и определите для него правило системных классов в загрузчике классов веб-приложения. Я рекомендую вам прочитать документацию о создании пользовательского модуля Jetty и о том, как определить правило классов сервера, следуя этой документации и этому примеру.
Обратите внимание, что в Servlet 4 вам это не понадобится, поскольку используемый API PushBuilder был стандартизирован в API Servlet."