001/* 002 * #%L 003 * HAPI FHIR - Server Framework 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.rest.api.server; 021 022import jakarta.annotation.Nonnull; 023import jakarta.annotation.Nullable; 024 025import java.io.Closeable; 026import java.io.IOException; 027import java.io.OutputStream; 028import java.io.Writer; 029import java.util.List; 030import java.util.Map; 031 032/** 033 * Implementations of this interface represent a response back to the client from the server. It is 034 * conceptually similar to {@link jakarta.servlet.http.HttpServletResponse} but intended to be agnostic 035 * of the server framework being used. 036 * <p> 037 * This class is a bit of an awkward abstraction given the two styles of servers it supports. 038 * Servlets work by writing to a servlet response that is provided as a parameter by the container. JAX-RS 039 * works by returning an object via a method return back to the containing framework. However using it correctly should 040 * make for compatible code across both approaches. 041 * </p> 042 */ 043public interface IRestfulResponse { 044 045 /** 046 * Initiate a new textual response. The Writer returned by this method must be finalized by 047 * calling {@link #commitResponse(Closeable)} later. 048 * <p> 049 * Note that the caller should not close the returned object, but should instead just 050 * return it to {@link #commitResponse(Closeable)} upon successful completion. This is 051 * different from normal Java practice where you would request it in a <code>try with resource</code> 052 * block, since in Servlets you are not actually required to close the writer/stream, and 053 * doing so automatically may prevent you from correctly handling exceptions. 054 * </p> 055 * 056 * @param theStatusCode The HTTP status code. 057 * @param theContentType The HTTP response content type. 058 * @param theCharset The HTTP response charset. 059 * @param theRespondGzip Should the response be GZip encoded? 060 * @return Returns a {@link Writer} that can accept the response body. 061 */ 062 @Nonnull 063 Writer getResponseWriter(int theStatusCode, String theContentType, String theCharset, boolean theRespondGzip) 064 throws IOException; 065 066 /** 067 * Initiate a new binary response. The OutputStream returned by this method must be finalized by 068 * calling {@link #commitResponse(Closeable)} later. This method should only be used for non-textual 069 * responses, for those use {@link #getResponseWriter(int, String, String, boolean)}. 070 * <p> 071 * Note that the caller should not close the returned object, but should instead just 072 * return it to {@link #commitResponse(Closeable)} upon successful completion. This is 073 * different from normal Java practice where you would request it in a <code>try with resource</code> 074 * block, since in Servlets you are not actually required to close the writer/stream, and 075 * doing so automatically may prevent you from correctly handling exceptions. 076 * </p> 077 * 078 * @param theStatusCode The HTTP status code. 079 * @param theContentType The HTTP response content type. 080 * @param theContentLength If known, the number of bytes that will be written. {@literal null} otherwise. 081 * @return Returns an {@link OutputStream} that can accept the response body. 082 */ 083 @Nonnull 084 OutputStream getResponseOutputStream(int theStatusCode, String theContentType, @Nullable Integer theContentLength) 085 throws IOException; 086 087 /** 088 * Finalizes the response streaming using the writer that was returned by calling either 089 * {@link #getResponseWriter(int, String, String, boolean)} or 090 * {@link #getResponseOutputStream(int, String, Integer)}. This method should only be 091 * called if the response writing/streaming actually completed successfully. If an error 092 * occurred you do not need to commit the response. 093 * 094 * @param theWriterOrOutputStream The {@link Writer} or {@link OutputStream} that was returned by this object, or a Writer/OutputStream 095 * which decorates the one returned by this object. 096 * @return If the server style requires a returned response object (i.e. JAX-RS Server), this method 097 * returns that object. If the server style does not require one (i.e. {@link ca.uhn.fhir.rest.server.RestfulServer}), 098 * this method returns {@literal null}. 099 */ 100 Object commitResponse(@Nonnull Closeable theWriterOrOutputStream) throws IOException; 101 102 /** 103 * Adds a response header. This method must be called prior to calling 104 * {@link #getResponseWriter(int, String, String, boolean)} or {@link #getResponseOutputStream(int, String, Integer)}. 105 * 106 * @param headerKey The header name 107 * @param headerValue The header value 108 */ 109 void addHeader(String headerKey, String headerValue); 110 111 /** 112 * Returns the headers added to this response 113 */ 114 Map<String, List<String>> getHeaders(); 115}