Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Prepare Pax Web version using JakartaEE 10+ API #1802

Open
grgrzybek opened this issue Nov 25, 2022 · 17 comments
Open

Prepare Pax Web version using JakartaEE 10+ API #1802

grgrzybek opened this issue Nov 25, 2022 · 17 comments
Assignees
Milestone

Comments

@grgrzybek grgrzybek added this to the 10.0.0 milestone Nov 25, 2022
@grgrzybek grgrzybek self-assigned this Nov 25, 2022
@grgrzybek
Copy link
Member Author

Changes in XSD (6.0):

  • new element /session-config/cookie-config/attribute (0..n) - SameSite Cookie Support jakartaee/servlet#175 (to implement SameSite for example)
  • new element /jsp-config/jsp-property-group/error-on-el-not-found (0..1)
  • new element /context-service
  • new element /managed-executor
  • new element /managed-scheduled-executor
  • new element /managed-thread-factory

@grgrzybek
Copy link
Member Author

No point of using org.osgi.service.servlet.runtime.dto.DTOConstants#FAILURE_REASON_SERVLET_WRITE_TO_LOCATION_DENIED - System.getSecurityManager() is prepared for removal after JDK17 - should be reflected in changed specification.

@grgrzybek
Copy link
Member Author

Jetty 11 seems to be only Servlet API 5.0.0... (so for example no jakarta.servlet.SessionCookieConfig#getAttributes() method).

@grgrzybek
Copy link
Member Author

Jetty 12 Beta (already available in Maven Central) is Servlet 6 implementation.

@grgrzybek grgrzybek changed the title Prepare Pax Web version using JakartaEE 9+ API Prepare Pax Web version using JakartaEE 10+ API Jun 13, 2023
@grgrzybek
Copy link
Member Author

grgrzybek commented Jun 15, 2023

(Dead)lock in HTTP/2 tests for Jetty (need to investigate writer closing in PUSH promises):

"jetty-qtp-24" #24 prio=5 os_prio=0 cpu=41,20ms elapsed=8,61s tid=0x00007f49b0aa3dc0 nid=0x1f701 waiting on condition  [0x00007f4955cfc000]
   java.lang.Thread.State: WAITING (parking)
	at jdk.internal.misc.Unsafe.park([email protected]/Native Method)
	- parking to wait for  <0x00000006127b1838> (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject)
	at java.util.concurrent.locks.LockSupport.park([email protected]/LockSupport.java:341)
	at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionNode.block([email protected]/AbstractQueuedSynchronizer.java:506)
	at java.util.concurrent.ForkJoinPool.unmanagedBlock([email protected]/ForkJoinPool.java:3463)
	at java.util.concurrent.ForkJoinPool.managedBlock([email protected]/ForkJoinPool.java:3434)
	at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await([email protected]/AbstractQueuedSynchronizer.java:1623)
	at org.eclipse.jetty.util.SharedBlockingCallback$Blocker.block(SharedBlockingCallback.java:214)
	at org.eclipse.jetty.ee10.servlet.HttpOutput.close(HttpOutput.java:571)
	at org.eclipse.jetty.ee10.servlet.writer.HttpWriter.close(HttpWriter.java:44)
	at org.eclipse.jetty.ee10.servlet.writer.ResponseWriter.close(ResponseWriter.java:158)
	- locked <0x00000006127b2788> (a org.eclipse.jetty.ee10.servlet.writer.Utf8HttpWriter)
	at org.ops4j.pax.web.service.jetty.internal.EmbeddedJettyHttps2Test$1.doGet(EmbeddedJettyHttps2Test.java:162)
	at jakarta.servlet.http.HttpServlet.service(HttpServlet.java:527)
	at jakarta.servlet.http.HttpServlet.service(HttpServlet.java:614)
	at org.eclipse.jetty.ee10.servlet.ServletHolder.handle(ServletHolder.java:739)
	at org.eclipse.jetty.ee10.servlet.ServletHandler$ChainEnd.doFilter(ServletHandler.java:1601)
	at org.eclipse.jetty.ee10.servlet.ServletHandler$MappedServlet.handle(ServletHandler.java:1534)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.lambda$handle$0(ServletChannel.java:425)
	at org.eclipse.jetty.ee10.servlet.ServletChannel$$Lambda$601/0x0000000800f32350.dispatch(Unknown Source)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.dispatch(ServletChannel.java:659)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.handle(ServletChannel.java:420)
	at org.eclipse.jetty.ee10.servlet.ServletHandler.handle(ServletHandler.java:458)
	at org.eclipse.jetty.server.handler.ContextHandler.handle(ContextHandler.java:819)
	at org.eclipse.jetty.server.Handler$Wrapper.handle(Handler.java:611)
	at org.eclipse.jetty.server.Server.handle(Server.java:175)
	at org.eclipse.jetty.server.internal.HttpChannelState$HandlerInvoker.run(HttpChannelState.java:553)
	at org.eclipse.jetty.util.thread.strategy.AdaptiveExecutionStrategy.runTask(AdaptiveExecutionStrategy.java:473)
	at org.eclipse.jetty.util.thread.strategy.AdaptiveExecutionStrategy.consumeTask(AdaptiveExecutionStrategy.java:436)
	at org.eclipse.jetty.util.thread.strategy.AdaptiveExecutionStrategy.tryProduce(AdaptiveExecutionStrategy.java:288)
	at org.eclipse.jetty.util.thread.strategy.AdaptiveExecutionStrategy.produce(AdaptiveExecutionStrategy.java:196)
	at org.eclipse.jetty.http2.HTTP2Connection.produce(HTTP2Connection.java:210)
	at org.eclipse.jetty.http2.HTTP2Connection.offerTask(HTTP2Connection.java:195)
	at org.eclipse.jetty.http2.server.internal.HTTP2ServerConnection.offerTask(HTTP2ServerConnection.java:345)
	at org.eclipse.jetty.http2.server.internal.HTTP2ServerConnection.push(HTTP2ServerConnection.java:227)
	at org.eclipse.jetty.http2.server.internal.HttpStreamOverHTTP2$2.succeeded(HttpStreamOverHTTP2.java:472)
	at org.eclipse.jetty.http2.server.internal.HttpStreamOverHTTP2$2.succeeded(HttpStreamOverHTTP2.java:468)
	at org.eclipse.jetty.util.Promise$Wrapper.succeeded(Promise.java:195)
	at org.eclipse.jetty.http2.HTTP2Session$1.succeeded(HTTP2Session.java:633)
	at org.eclipse.jetty.http2.HTTP2Session$1.succeeded(HTTP2Session.java:626)
	at org.eclipse.jetty.http2.HTTP2Session$StreamsState.lambda$createLocalStream$20(HTTP2Session.java:2182)
	at org.eclipse.jetty.http2.HTTP2Session$StreamsState$$Lambda$605/0x0000000800f34fc0.run(Unknown Source)
	at org.eclipse.jetty.util.Callback$3.succeeded(Callback.java:161)
	at org.eclipse.jetty.util.Callback$Nested.succeeded(Callback.java:399)
	at org.eclipse.jetty.http2.HTTP2Session$ControlEntry.succeeded(HTTP2Session.java:1381)
	at org.eclipse.jetty.http2.internal.HTTP2Flusher$$Lambda$595/0x0000000800f179d8.accept(Unknown Source)
	at java.util.ArrayList.forEach([email protected]/ArrayList.java:1511)
	at org.eclipse.jetty.http2.internal.HTTP2Flusher.finish(HTTP2Flusher.java:323)
	at org.eclipse.jetty.http2.internal.HTTP2Flusher.succeeded(HTTP2Flusher.java:315)
	at org.eclipse.jetty.io.WriteFlusher.write(WriteFlusher.java:291)
	at org.eclipse.jetty.io.WriteFlusher.write(WriteFlusher.java:254)
	at org.eclipse.jetty.io.AbstractEndPoint.write(AbstractEndPoint.java:374)
	at org.eclipse.jetty.http2.internal.HTTP2Flusher.process(HTTP2Flusher.java:292)
	at org.eclipse.jetty.util.IteratingCallback.processing(IteratingCallback.java:243)
	at org.eclipse.jetty.util.IteratingCallback.iterate(IteratingCallback.java:224)
	at org.eclipse.jetty.http2.HTTP2Session$StreamsState.flush(HTTP2Session.java:2301)
	at org.eclipse.jetty.http2.HTTP2Session$StreamsState.createLocalStream(HTTP2Session.java:2199)
	at org.eclipse.jetty.http2.HTTP2Session$StreamsState.push(HTTP2Session.java:2164)
	at org.eclipse.jetty.http2.HTTP2Session.push(HTTP2Session.java:625)
	at org.eclipse.jetty.http2.HTTP2Stream.push(HTTP2Stream.java:152)
	at org.eclipse.jetty.http2.server.internal.HttpStreamOverHTTP2.push(HttpStreamOverHTTP2.java:467)
	at org.eclipse.jetty.server.internal.HttpChannelState$ChannelRequest.push(HttpChannelState.java:935)
	at org.eclipse.jetty.server.Request$Wrapper.push(Request.java:655)
	at org.eclipse.jetty.server.Request$Wrapper.push(Request.java:655)
	at org.eclipse.jetty.ee10.servlet.PushBuilderImpl.push(PushBuilderImpl.java:125)
	at org.ops4j.pax.web.service.jetty.internal.EmbeddedJettyHttps2Test$1.doGet(EmbeddedJettyHttps2Test.java:173)
	at jakarta.servlet.http.HttpServlet.service(HttpServlet.java:527)
	at jakarta.servlet.http.HttpServlet.service(HttpServlet.java:614)
	at org.eclipse.jetty.ee10.servlet.ServletHolder.handle(ServletHolder.java:739)
	at org.eclipse.jetty.ee10.servlet.ServletHandler$ChainEnd.doFilter(ServletHandler.java:1601)
	at org.eclipse.jetty.ee10.servlet.ServletHandler$MappedServlet.handle(ServletHandler.java:1534)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.lambda$handle$0(ServletChannel.java:425)
	at org.eclipse.jetty.ee10.servlet.ServletChannel$$Lambda$601/0x0000000800f32350.dispatch(Unknown Source)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.dispatch(ServletChannel.java:659)
	at org.eclipse.jetty.ee10.servlet.ServletChannel.handle(ServletChannel.java:420)
	at org.eclipse.jetty.ee10.servlet.ServletHandler.handle(ServletHandler.java:458)
	at org.eclipse.jetty.server.handler.ContextHandler.handle(ContextHandler.java:819)
	at org.eclipse.jetty.server.Handler$Wrapper.handle(Handler.java:611)
	at org.eclipse.jetty.server.Server.handle(Server.java:175)
	at org.eclipse.jetty.server.internal.HttpChannelState$HandlerInvoker.run(HttpChannelState.java:553)
	at org.eclipse.jetty.util.thread.Invocable$ReadyTask.run(Invocable.java:105)
	at org.eclipse.jetty.http2.server.internal.HttpStreamOverHTTP2$1.run(HttpStreamOverHTTP2.java:136)
	at org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:969)
	at org.eclipse.jetty.util.thread.QueuedThreadPool$Runner.doRunJob(QueuedThreadPool.java:1194)
	at org.eclipse.jetty.util.thread.QueuedThreadPool$Runner.run(QueuedThreadPool.java:1149)
	at java.lang.Thread.run([email protected]/Thread.java:833)

grgrzybek added a commit that referenced this issue Jun 21, 2023
grgrzybek added a commit that referenced this issue Jun 21, 2023
@grgrzybek
Copy link
Member Author

Changes between 5 and 6 (interfaces):

  • added jakarta.servlet.descriptor.JspPropertyGroupDescriptor.getErrorOnELNotFound()
  • deprecated jakarta.servlet.http.Cookie.set|getComment()
  • deprecated jakarta.servlet.http.Cookie.set|getVersion()
  • added jakarta.servlet.http.Cookie.setAttribute()
  • added jakarta.servlet.http.Cookie.getAttribute()
  • added jakarta.servlet.http.Cookie.setAttributes()
  • removed jakarta.servlet.http.HttpServletRequest.isRequestedSessionIdFromUrl() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpServletRequestWrapper.isRequestedSessionIdFromUrl() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpServletResponse.encodeUrl() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpServletResponse.encodeRedirectUrl() (deprecated in version 5. encodeRedirectURL() should be used)
  • removed jakarta.servlet.http.HttpServletResponse.setStatus() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpServletResponseWrapper.encodeUrl() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpServletResponseWrapper.encodeRedirectUrl() (deprecated in version 5. encodeRedirectURL() should be used)
  • removed jakarta.servlet.http.HttpServletResponseWrapper.setStatus() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSession.getSessionContext() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSession.getValue() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSession.getValueNames() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSession.putValue() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSession.removeValue() (deprecated in version 5)
  • removed jakarta.servlet.http.HttpSessionContext class (deprecated in version 5)
  • removed jakarta.servlet.http.HttpUtils class (deprecated in version 5)
  • added jakarta.servlet.ServletConnection interface
  • removed jakarta.servlet.ServletContext.getServlet() (deprecated in version 5)
  • removed jakarta.servlet.ServletContext.getServlets() (deprecated in version 5)
  • removed jakarta.servlet.ServletContext.getServletNames() (deprecated in version 5)
  • removed jakarta.servlet.ServletContext.log() (deprecated in version 5)
  • removed jakarta.servlet.ServletRequest.getRealPath() (deprecated in version 5)
  • added jakarta.servlet.ServletContext.getRequestId()
  • added jakarta.servlet.ServletContext.getProtocolRequestId()
  • added jakarta.servlet.ServletContext.getServletConnection()
  • removed jakarta.servlet.ServletRequestWrapper.getRealPath() (deprecated in version 5)
  • added jakarta.servlet.ServletContextWrapper.getRequestId()
  • added jakarta.servlet.ServletContextWrapper.getProtocolRequestId()
  • added jakarta.servlet.ServletContextWrapper.getServletConnection()
  • deprecated jakarta.servlet.SessionCookieConfig.set|getComment() (no longer required by RFC 6265)
  • added jakarta.servlet.SessionCookieConfig.setAttribute()
  • added jakarta.servlet.SessionCookieConfig.getAttribute()
  • added jakarta.servlet.SessionCookieConfig.getAttributes()
  • removed jakarta.servlet.SingleThreadModel interface
  • removed jakarta.servlet.UnavailableException constructors that accept Servlet paramater (deprecated in version 5)
  • removed jakarta.servlet.UnavailableException.getServlet() (deprecated in version 5)

Changes between 5 and 6 (documentation):

  • jakarta.servlet.annotation.HttpMethodConstraint.value() mentions now RFC 7231 instead of RFC 2616 (same for several other classes)
  • jakarta.servlet.http.Cookie.Cookie() mentions now RFC 6265 instead of RFC 2109 (same for several other classes)

Changes between 5 and 6 (implementation):

  • jakarta.servlet.http.HttpServlet.doHead() - HEAD without body is handled only in legacy mode (jakarta.servlet.http.legacyDoHead init param). By default, doGet() is called

Changes between 5 and 6 (other):

  • jakarta.servlet.http.HttpServlet - added jakarta.servlet.http.legacyDoHead option
  • added jakarta/servlet/resources/jakartaee_10.xsd
  • added jakarta/servlet/resources/jsp_3_1.xsd
  • added jakarta/servlet/resources/web-app_6_0.xsd
  • added jakarta/servlet/resources/web-common_6_0.xsd
  • added jakarta/servlet/resources/web-fragment_6_0.xsd
  • added jakarta/servlet/resources/web-jsptaglibrary_3_1.xsd

@grgrzybek
Copy link
Member Author

grgrzybek commented Nov 27, 2023

Another problem with Jetty 12 is the use of java.nio.file.Paths#get(java.net.URI) which fails for jar:, bundle: because the underlying URL handler is not used. We have this exception:

11:44:56.835 [paxweb-config-1-thread-1 (change controller)] ERROR (Activator.java:623) org.ops4j.pax.web.service.internal.Activator - Unable to start Pax Web server: Provider "bundle" not installed
java.nio.file.FileSystemNotFoundException: Provider "bundle" not installed
	at java.nio.file.Path.of(Path.java:212) ~[?:?]
	at java.nio.file.Paths.get(Paths.java:98) ~[?:?]
	at org.eclipse.jetty.util.resource.PathResourceFactory.newResource(PathResourceFactory.java:26) ~[jetty-util-12.0.3.jar:12.0.3]
	at org.eclipse.jetty.util.resource.ResourceFactory.newResource(ResourceFactory.java:370) ~[jetty-util-12.0.3.jar:12.0.3]
	at org.ops4j.pax.web.service.jetty.internal.JettyFactory.newResource(JettyFactory.java:588) ~[pax-web-jetty-10.0.0-SNAPSHOT.jar:?]
	at org.ops4j.pax.web.service.jetty.internal.JettyServerWrapper.applyJettyConfiguration(JettyServerWrapper.java:371) ~[pax-web-jetty-10.0.0-SNAPSHOT.jar:?]
	at org.ops4j.pax.web.service.jetty.internal.JettyServerWrapper.configure(JettyServerWrapper.java:272) ~[pax-web-jetty-10.0.0-SNAPSHOT.jar:?]
	at org.ops4j.pax.web.service.jetty.internal.JettyServerController.configure(JettyServerController.java:94) ~[pax-web-jetty-10.0.0-SNAPSHOT.jar:?]
	at org.ops4j.pax.web.service.internal.Activator.performConfiguration(Activator.java:552) ~[?:?]
...

or

java.nio.file.FileSystemNotFoundException
	at jdk.zipfs/jdk.nio.zipfs.ZipFileSystemProvider.getFileSystem(ZipFileSystemProvider.java:173)
	at jdk.zipfs/jdk.nio.zipfs.ZipFileSystemProvider.getPath(ZipFileSystemProvider.java:159)
	at java.base/java.nio.file.Path.of(Path.java:208)
	at java.base/java.nio.file.Paths.get(Paths.java:98)
	at org.eclipse.jetty.util.resource.PathResourceFactory.newResource(PathResourceFactory.java:26)
	at org.eclipse.jetty.util.resource.ResourceFactory.newResource(ResourceFactory.java:370)
	at org.ops4j.pax.web.service.jetty.internal.JettyFactory.newResource(JettyFactory.java:588)
	at org.ops4j.pax.web.service.jetty.internal.JettyServerWrapper.applyJettyConfiguration(JettyServerWrapper.java:371)
	at org.ops4j.pax.web.service.jetty.internal.JettyServerWrapper.configure(JettyServerWrapper.java:272)
	at org.ops4j.pax.web.service.jetty.internal.JettyServerController.configure(JettyServerController.java:94)
	at org.ops4j.pax.web.itest.server.controller.ServerControllerBasicConfigurationTest.justInstantiateWithoutOsgi(ServerControllerBasicConfigurationTest.java:80)
...

Similar problem is described under spring-projects/spring-boot#7161

@d0mmi
Copy link

d0mmi commented Mar 14, 2024

Are there any news on this?

@grgrzybek
Copy link
Member Author

Well, at some point I created very crude implementation of pax-web-extender-whiteboard which implements https://docs.osgi.org/specification/osgi.cmpn/8.1.0/service.servlet.html and it worked with Jolokia 2.
But I didn't have much time to finish the implementation.

Last week Jetty 12.0.7 was released and I was going to get back to this issue. Maybe next week I'll revisit it...

@ecki
Copy link
Contributor

ecki commented Nov 28, 2024

Is there a way we can support jakarta. API development of this (especially for Tommcat). We would like to see it in Karaf.

@grgrzybek
Copy link
Member Author

There is a way, but unfortunately it's a long way. while other JakartaEE implementations are ready, their OSGi support is far from perfect...
And I didn't have much time recently for progressing with this.

And the branch https://github.com/ops4j/org.ops4j.pax.web/commits/main-jakarta/ is open for PRs, so if anyone could contribute, it'd be great ;)

@paulrutter
Copy link

paulrutter commented Dec 10, 2024

Is there a way we can support jakarta. API development of this (especially for Tommcat). We would like to see it in Karaf.

Apache Felix offers a jakarta implementation of the OSGi spec as well, based on Jetty12. https://github.com/apache/felix-dev/tree/master/http/jetty12

@grgrzybek is pax offering other / more features on top of what Felix offers?

@grgrzybek
Copy link
Member Author

Pax Web's history is long - it started when first HttpService OSGi CMPN was created.

The main goal of Pax Web was always to provide choice of the runtime covered by OSGi CMPN specification. So while Felix HTTP implements OSGi CMPN perfectly on top of Jetty using single servlet, Pax Web deals with three different Servlet API implementations:

  • Jetty
  • Tomcat
  • Undertow

And each tracked servlet/filter/listener/error-page/welcome-file/servlet-container-initializer/websocket/jsp/... is configured within one of these runtimes in more canonical way - not proxied by single dispatcher servlet, but added directly to Jetty/Tomcat/Unertow context (Jetty's ServletContextHandler, Tomcat's StandardContext and Undertow's Deployment).

Because we want to have unified experience, there is special code that make it possible. Also we've found some spec-violations like:

Also - OSGi CMPN Whiteboard Specificaion supports: servlets (including resources and error pages), filters and listeners. Pax Web (I know that Felix HTTP supports some additional web elements as well) supports additionally JSPs, welcome pages, websockets, servlet container listeners, security mappings, login configs (everything you can find in web.xml).

So treat Pax Web as alternative implementation, where you can use Tomcat and/or Undertow.
Apache Karaf uses (at least up to 4.4.x) Pax Web. Also Red Hat Fuse based on Karaf uses Pax Web (with Undertow) as default implementation.

@paulrutter
Copy link

Thanks for the explanation. I just wanted to point out that Felix HTTP offers a jakartified version based on Jetty12 (including optional websocket support), but didn't know the exact differences compared to Pax Web. That's clear now.
I understand supporting Jakarta for multiple Servlet API implementations is harder than just supporting one.

@grgrzybek
Copy link
Member Author

If we consider 3 deployment methods:

Then multiple Jakarta versions are easiest in Whiteboard mode (though Pax Web doesn't do it yet) - even on single target runtime (without Jetty 12 magic that combines Javax and Jakarta).
Support for multi-specification WABs is more problematic and for HttpService we'd simply need two services.

@dstoch
Copy link

dstoch commented Jan 21, 2025

Jetty 9, 10, 11 have end-of-life 2025.01:
jetty/jetty.project#7958
jetty/jetty.project#10485

So for now the only supported version is Jetty 12, which if I see properly, supports both
old Javax API (EE8) and the new Jakarta API.

Would it be possible to prepare a version of PaxWeb with Jetty 12 with old Javax API, based on PaxWeb 9.x?

@grgrzybek
Copy link
Member Author

Would it be possible to prepare a version of PaxWeb with Jetty 12 with old Javax API, based on PaxWeb 9.x?

I guess at some point this will be the only option... Thanks for making me realize this ;)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

5 participants