Отдельный 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."

Другие вопросы по тегам