All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.luminis.quic.client.h09.Http09Client Maven / Gradle / Ivy

/*
 * Copyright © 2021, 2022, 2023, 2024 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.QuicConnection;
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
 * 
    *
  • https://www.w3.org/Protocols/HTTP/AsImplemented.html
  • *
  • https://superuser.com/questions/1504500/what-is-http-0-9-request
  • *
*/ public class Http09Client extends HttpClient { private final QuicClientConnection quicConnection; private final boolean with0RTT; 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() == QuicConnection.QuicVersion.V1 || quicConnection.getQuicVersion() == QuicConnection.QuicVersion.V2) { alpn = "hq-interop"; } else { String draftVersion = "34"; alpn = "hq-" + draftVersion; } if (with0RTT) { QuicClientConnection.StreamEarlyData earlyData = new QuicClientConnection.StreamEarlyData(httpGetCommand.getBytes(), true); httpStream = quicConnection.connect(List.of(earlyData)).get(0); } else { quicConnection.connect(); } } 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); } } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy