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

org.eclipse.jetty.reactive.client.ReactiveResponse Maven / Gradle / Ivy

/*
 * Copyright (c) 2017-2022 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.eclipse.jetty.reactive.client;

import java.nio.ByteBuffer;
import java.util.Locale;
import java.util.function.BiFunction;

import org.eclipse.jetty.client.api.Response;
import org.eclipse.jetty.http.HttpFields;
import org.eclipse.jetty.http.HttpHeader;
import org.eclipse.jetty.reactive.client.internal.BufferingProcessor;
import org.eclipse.jetty.reactive.client.internal.DiscardingProcessor;
import org.reactivestreams.Publisher;

/**
 * 

A reactive wrapper over Jetty's {@code HttpClient} {@link Response}.

*

A ReactiveResponse is available as soon as the response headers arrived * to the client. The response content is processed by a response content * function specified in {@link ReactiveRequest#response(BiFunction)}.

*/ public class ReactiveResponse { private final ReactiveRequest request; private final Response response; private String mediaType; private String encoding; public ReactiveResponse(ReactiveRequest request, Response response) { this.request = request; this.response = response; } /** * @return the ReactiveRequest correspondent to this response */ public ReactiveRequest getReactiveRequest() { return request; } /** * @return the wrapped Jetty response */ public Response getResponse() { return response; } /** * @return the HTTP status code */ public int getStatus() { return response.getStatus(); } /** * @return the HTTP response headers */ public HttpFields getHeaders() { return response.getHeaders(); } /** * @return the media type specified by the {@code Content-Type} header * @see #getEncoding() */ public String getMediaType() { resolveContentType(); return mediaType.isEmpty() ? null : mediaType; } /** * @return the encoding specified by the {@code Content-Type} header * @see #getMediaType() */ public String getEncoding() { resolveContentType(); return encoding.isEmpty() ? null : encoding; } private void resolveContentType() { if (mediaType == null) { String contentType = getHeaders().get(HttpHeader.CONTENT_TYPE); if (contentType != null) { String media = contentType; String charset = "charset="; int index = contentType.toLowerCase(Locale.ENGLISH).indexOf(charset); if (index > 0) { media = contentType.substring(0, index); String encoding = contentType.substring(index + charset.length()); // Sometimes charsets arrive with an ending semicolon int semicolon = encoding.indexOf(';'); if (semicolon > 0) { encoding = encoding.substring(0, semicolon).trim(); } this.encoding = encoding; } else { this.encoding = ""; } int semicolon = media.indexOf(';'); if (semicolon > 0) { media = media.substring(0, semicolon).trim(); } this.mediaType = media; } else { this.mediaType = ""; this.encoding = ""; } } } @Override public String toString() { return String.format("Reactive[%s]", response); } /** * Collects utility methods to process response content. */ public static class Content { /** * @return a response content processing function that discards the content */ public static BiFunction, Publisher> discard() { return (response, content) -> { DiscardingProcessor result = new DiscardingProcessor(response); content.subscribe(result); return result; }; } /** * @return a response content processing function that converts the content to a string */ public static BiFunction, Publisher> asString() { return (response, content) -> { BufferingProcessor result = new BufferingProcessor(response); content.subscribe(result); return result; }; } } public static class Event { private final Type type; private final ReactiveResponse response; private final ByteBuffer content; private final Throwable failure; public Event(Type type, ReactiveResponse response) { this(type, response, null, null); } public Event(Type type, ReactiveResponse response, ByteBuffer content) { this(type, response, content, null); } public Event(Type type, ReactiveResponse response, Throwable failure) { this(type, response, null, failure); } private Event(Type type, ReactiveResponse response, ByteBuffer content, Throwable failure) { this.type = type; this.response = response; this.content = content; this.failure = failure; } /** * @return the event type */ public Type getType() { return type; } /** * @return the response that generated this event */ public ReactiveResponse getResponse() { return response; } /** * @return the event content, or null if this is not a content event */ public ByteBuffer getContent() { return content; } /** * @return the event failure, or null if this is not a failure event */ public Throwable getFailure() { return failure; } public enum Type { BEGIN, HEADERS, CONTENT, SUCCESS, FAILURE, COMPLETE } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy