diff --git a/.idea/encodings.xml b/.idea/encodings.xml index de5572116383..ca018ebc3ab9 100644 --- a/.idea/encodings.xml +++ b/.idea/encodings.xml @@ -32,6 +32,9 @@ + + + diff --git a/Jenkinsfile b/Jenkinsfile index 8b87983f43fc..6757e46642b9 100644 --- a/Jenkinsfile +++ b/Jenkinsfile @@ -14,7 +14,7 @@ properties([ def axes = [ platforms: ['linux', 'windows'], - jdks: [11, 17, 21], + jdks: [17, 21], ] stage('Record build') { diff --git a/pom.xml b/pom.xml index 6293399fbd82..faf0941d0f37 100644 --- a/pom.xml +++ b/pom.xml @@ -53,6 +53,7 @@ THE SOFTWARE. bom websocket/spi websocket/jetty10 + websocket/jetty12-ee8 core war test @@ -99,7 +100,8 @@ THE SOFTWARE. 1.29 false - 6.19 + 7.0-rc912.f5b_c197b_06e7 + 17 + + 4.0.0 + + + org.jenkins-ci.main + jenkins-parent + ${revision}${changelist} + ../.. + + + websocket-jetty12-ee8 + Jetty 12 (EE 8) implementation for WebSocket + An implementation of the WebSocket handler that works with Jetty 12 (EE 8). + + + + + org.jenkins-ci.main + jenkins-bom + ${project.version} + pom + import + + + + + + + org.jenkins-ci + winstone + ${winstone.version} + true + + + org.jenkins-ci.main + websocket-spi + ${project.version} + + + org.kohsuke + access-modifier-annotation + + + org.kohsuke.metainf-services + metainf-services + true + + + + + + + org.apache.maven.plugins + maven-javadoc-plugin + + true + + + + + diff --git a/websocket/jetty12-ee8/src/main/java/jenkins/websocket/Jetty12EE8Provider.java b/websocket/jetty12-ee8/src/main/java/jenkins/websocket/Jetty12EE8Provider.java new file mode 100644 index 000000000000..0fe0fef8ac93 --- /dev/null +++ b/websocket/jetty12-ee8/src/main/java/jenkins/websocket/Jetty12EE8Provider.java @@ -0,0 +1,179 @@ +/* + * The MIT License + * + * Copyright 2022 CloudBees, Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ + +package jenkins.websocket; + +import java.io.IOException; +import java.nio.ByteBuffer; +import java.time.Duration; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.Future; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import org.eclipse.jetty.ee8.websocket.api.Session; +import org.eclipse.jetty.ee8.websocket.api.WebSocketListener; +import org.eclipse.jetty.ee8.websocket.api.WriteCallback; +import org.eclipse.jetty.ee8.websocket.server.JettyServerUpgradeRequest; +import org.eclipse.jetty.ee8.websocket.server.JettyServerUpgradeResponse; +import org.eclipse.jetty.ee8.websocket.server.JettyWebSocketServerContainer; +import org.kohsuke.MetaInfServices; +import org.kohsuke.accmod.Restricted; +import org.kohsuke.accmod.restrictions.NoExternalUse; + +@Restricted(NoExternalUse.class) +@MetaInfServices(Provider.class) +public class Jetty12EE8Provider implements Provider { + + /** + * Number of seconds a WebsocketConnection may stay idle until it expires. + * Zero to disable. + * This value must be higher than the jenkins.websocket.pingInterval. + * Per Jetty 12 documentation + * a ping mechanism should keep the websocket active. Therefore, the idle timeout must be higher than the ping + * interval to avoid timeout issues. + */ + private static long IDLE_TIMEOUT_SECONDS = Long.getLong("jenkins.websocket.idleTimeout", 60L); + + private static final String ATTR_LISTENER = Jetty12EE8Provider.class.getName() + ".listener"; + + private boolean initialized = false; + + public Jetty12EE8Provider() { + JettyWebSocketServerContainer.class.hashCode(); + } + + private void init(HttpServletRequest req) { + if (!initialized) { + JettyWebSocketServerContainer.getContainer(req.getServletContext()).setIdleTimeout(Duration.ofSeconds(IDLE_TIMEOUT_SECONDS)); + initialized = true; + } + } + + @Override + public Handler handle(HttpServletRequest req, HttpServletResponse rsp, Listener listener) throws Exception { + init(req); + req.setAttribute(ATTR_LISTENER, listener); + // TODO Jetty 12 has no obvious equivalent to WebSocketServerFactory.isUpgradeRequest; RFC6455Negotiation? + if (!"websocket".equalsIgnoreCase(req.getHeader("Upgrade"))) { + rsp.sendError(HttpServletResponse.SC_BAD_REQUEST, "only WS connections accepted here"); + return null; + } + if (!JettyWebSocketServerContainer.getContainer(req.getServletContext()).upgrade(Jetty12EE8Provider::createWebSocket, req, rsp)) { + rsp.sendError(HttpServletResponse.SC_BAD_REQUEST, "did not manage to upgrade"); + return null; + } + return new Handler() { + @Override + public Future sendBinary(ByteBuffer data) throws IOException { + CompletableFuture f = new CompletableFuture<>(); + session().getRemote().sendBytes(data, new WriteCallbackImpl(f)); + return f; + } + + @Override + public void sendBinary(ByteBuffer partialByte, boolean isLast) throws IOException { + session().getRemote().sendPartialBytes(partialByte, isLast); + } + + @Override + public Future sendText(String text) throws IOException { + CompletableFuture f = new CompletableFuture<>(); + session().getRemote().sendString(text, new WriteCallbackImpl(f)); + return f; + } + + @Override + public Future sendPing(ByteBuffer applicationData) throws IOException { + CompletableFuture f = new CompletableFuture<>(); + session().getRemote().sendPing(applicationData, new WriteCallbackImpl(f)); + return f; + } + + @Override + public void close() throws IOException { + session().close(); + } + + private Session session() { + Session session = (Session) listener.getProviderSession(); + if (session == null) { + throw new IllegalStateException("missing session"); + } + return session; + } + }; + } + + private static final class WriteCallbackImpl implements WriteCallback { + private final CompletableFuture f; + + WriteCallbackImpl(CompletableFuture f) { + this.f = f; + } + + @Override + public void writeSuccess() { + f.complete(null); + } + + @Override + public void writeFailed(Throwable x) { + f.completeExceptionally(x); + } + } + + private static Object createWebSocket(JettyServerUpgradeRequest req, JettyServerUpgradeResponse resp) { + Listener listener = (Listener) req.getHttpServletRequest().getAttribute(ATTR_LISTENER); + if (listener == null) { + throw new IllegalStateException("missing listener attribute"); + } + return new WebSocketListener() { + @Override + public void onWebSocketBinary(byte[] payload, int offset, int length) { + listener.onWebSocketBinary(payload, offset, length); + } + + @Override + public void onWebSocketText(String message) { + listener.onWebSocketText(message); + } + + @Override + public void onWebSocketClose(int statusCode, String reason) { + listener.onWebSocketClose(statusCode, reason); + } + + @Override + public void onWebSocketConnect(Session session) { + listener.onWebSocketConnect(session); + } + + @Override + public void onWebSocketError(Throwable cause) { + listener.onWebSocketError(cause); + } + }; + } + +}