/* * Copyright © 2021, 2022, 2023 Peter Doornbosch * * This file is part of Kwik, an implementation of the QUIC protocol in Java. * * Kwik is free software: you can redistribute it and/or modify it under * the terms of the GNU Lesser General Public License as published by the * Free Software Foundation, either version 3 of the License, or (at your option) * any later version. * * Kwik is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for * more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program. If not, see . */ package net.luminis.quic.client.h09; import net.luminis.quic.QuicClientConnection; import net.luminis.quic.QuicStream; import net.luminis.quic.concurrent.DaemonThreadFactory; import javax.net.ssl.SSLContext; import javax.net.ssl.SSLParameters; import java.io.IOException; import java.io.InputStream; import java.net.Authenticator; import java.net.CookieHandler; import java.net.ProxySelector; import java.net.http.HttpClient; import java.net.http.HttpHeaders; import java.net.http.HttpRequest; import java.net.http.HttpResponse; import java.nio.ByteBuffer; import java.time.Duration; import java.util.Collections; import java.util.List; import java.util.Optional; import java.util.concurrent.*; /** * A HTTP client for HTTP 0.9 requests. * See * */ public class Http09Client extends HttpClient { private final QuicClientConnection quicConnection; private final boolean with0RTT; private final int connectionTimeout = 10_000; private final ExecutorService executorService; public Http09Client(QuicClientConnection quicConnection, boolean with0RTT) { this.quicConnection = quicConnection; this.with0RTT = with0RTT; executorService = Executors.newCachedThreadPool(new DaemonThreadFactory("http09")); } @Override public Optional cookieHandler() { return Optional.empty(); } @Override public Optional connectTimeout() { return Optional.empty(); } @Override public Redirect followRedirects() { return null; } @Override public Optional proxy() { return Optional.empty(); } @Override public SSLContext sslContext() { return null; } @Override public SSLParameters sslParameters() { return null; } @Override public Optional authenticator() { return Optional.empty(); } @Override public Version version() { return null; } @Override public Optional executor() { return Optional.empty(); } @Override public HttpResponse send(HttpRequest request, HttpResponse.BodyHandler responseBodyHandler) throws IOException, InterruptedException { AsyncHttpRequest requestTask = new AsyncHttpRequest<>(request, responseBodyHandler); requestTask.run(); try { return requestTask.get(); } catch (ExecutionException e) { if (e.getCause() instanceof IOException) { throw (IOException) e.getCause(); } else { throw new IOException(e); } } } @Override public CompletableFuture> sendAsync(HttpRequest request, HttpResponse.BodyHandler responseBodyHandler) { AsyncHttpRequest future = new AsyncHttpRequest<>(request, responseBodyHandler); future.start(); return future; } @Override public CompletableFuture> sendAsync(HttpRequest request, HttpResponse.BodyHandler responseBodyHandler, HttpResponse.PushPromiseHandler pushPromiseHandler) { // HTTP 0.9 does not support server push throw new UnsupportedOperationException(); } private class AsyncHttpRequest extends CompletableFuture> { private final HttpRequest request; private final HttpResponse.BodyHandler responseBodyHandler; private QuicStream httpStream; public AsyncHttpRequest(HttpRequest request, HttpResponse.BodyHandler responseBodyHandler) { this.request = request; this.responseBodyHandler = responseBodyHandler; } /** * Run this request asynchronously. */ public void start() { executorService.submit(() -> run()); } /** * Run this request synchronously. */ public void run() { try { complete(send(request, responseBodyHandler)); } catch (IOException | RuntimeException | InterruptedException ex) { completeExceptionally(ex); } catch (Exception ex) { completeExceptionally(ex); } } @Override public boolean cancel(boolean mayInterruptIfRunning) { httpStream.closeInput(9); // HTTP 0.9 does not define error codes for QUIC streams return super.cancel(mayInterruptIfRunning); } private HttpResponse send(HttpRequest request, HttpResponse.BodyHandler responseBodyHandler) throws IOException, InterruptedException { String requestPath = request.uri().getPath(); String httpGetCommand = "GET " + requestPath + "\r\n"; httpStream = null; if (!quicConnection.isConnected()) { String alpn; if (quicConnection.getQuicVersion().isV1V2()) { alpn = "hq-interop"; } else { String draftVersion = quicConnection.getQuicVersion().getDraftVersion(); alpn = "hq-" + draftVersion; } if (with0RTT) { QuicClientConnection.StreamEarlyData earlyData = new QuicClientConnection.StreamEarlyData(httpGetCommand.getBytes(), true); httpStream = quicConnection.connect(connectionTimeout, alpn, null, List.of(earlyData)).get(0); } else { quicConnection.connect(connectionTimeout, alpn, null, null); } } if (httpStream == null) { boolean bidirectional = true; httpStream = quicConnection.createStream(bidirectional); httpStream.getOutputStream().write(httpGetCommand.getBytes()); httpStream.getOutputStream().close(); } HttpResponse.BodySubscriber bodySubscriber = responseBodyHandler.apply(new HttpResponse.ResponseInfo() { @Override public int statusCode() { return 200; } @Override public HttpHeaders headers() { return HttpHeaders.of(Collections.emptyMap(), (u, v) -> true); } @Override public Version version() { return null; } }); bodySubscriber.onSubscribe(new Flow.Subscription() { @Override public void request(long n) {} @Override public void cancel() {} }); InputStream inputStream = httpStream.getInputStream(); while (true) { int available = inputStream.available(); if (available == 0) { available = 1; // Wait for more data } byte[] buffer = new byte[available]; int read = inputStream.read(buffer); if (read < 0) { break; } bodySubscriber.onNext(List.of(ByteBuffer.wrap(buffer, 0, read))); } bodySubscriber.onComplete(); try { T body = bodySubscriber.getBody().toCompletableFuture().get(); return new HttpResponseImpl(request, body); } catch (ExecutionException e) { throw new IOException(e); } } } }