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.server.servlet;
021
022import ca.uhn.fhir.rest.api.Constants;
023import ca.uhn.fhir.rest.server.BaseRestfulResponse;
024import ca.uhn.fhir.util.IoUtil;
025import jakarta.annotation.Nonnull;
026import jakarta.servlet.ServletOutputStream;
027import jakarta.servlet.http.HttpServletResponse;
028import org.apache.commons.lang3.StringUtils;
029import org.apache.commons.lang3.Validate;
030
031import java.io.Closeable;
032import java.io.IOException;
033import java.io.OutputStream;
034import java.io.OutputStreamWriter;
035import java.io.Writer;
036import java.nio.charset.StandardCharsets;
037import java.util.List;
038import java.util.Map.Entry;
039import java.util.zip.GZIPOutputStream;
040
041public class ServletRestfulResponse extends BaseRestfulResponse<ServletRequestDetails> {
042
043        private Writer myWriter;
044        private OutputStream myOutputStream;
045
046        /**
047         * Constructor
048         */
049        public ServletRestfulResponse(ServletRequestDetails servletRequestDetails) {
050                super(servletRequestDetails);
051        }
052
053        @Nonnull
054        @Override
055        public OutputStream getResponseOutputStream(int theStatusCode, String theContentType, Integer theContentLength)
056                        throws IOException {
057                Validate.isTrue(myWriter == null, "getResponseOutputStream() called multiple times");
058                Validate.isTrue(myOutputStream == null, "getResponseOutputStream() called after getResponseWriter()");
059
060                addHeaders();
061                HttpServletResponse httpResponse = getRequestDetails().getServletResponse();
062                httpResponse.setStatus(theStatusCode);
063                httpResponse.setContentType(theContentType);
064                httpResponse.setCharacterEncoding(null);
065                if (theContentLength != null) {
066                        httpResponse.setContentLength(theContentLength);
067                }
068                myOutputStream = httpResponse.getOutputStream();
069                return myOutputStream;
070        }
071
072        @Nonnull
073        @Override
074        public Writer getResponseWriter(int theStatusCode, String theContentType, String theCharset, boolean theRespondGzip)
075                        throws IOException {
076                Validate.isTrue(myOutputStream == null, "getResponseWriter() called after getResponseOutputStream()");
077
078                addHeaders();
079                HttpServletResponse theHttpResponse = getRequestDetails().getServletResponse();
080                theHttpResponse.setCharacterEncoding(theCharset);
081                theHttpResponse.setStatus(theStatusCode);
082                theHttpResponse.setContentType(theContentType);
083                if (theRespondGzip) {
084                        theHttpResponse.addHeader(Constants.HEADER_CONTENT_ENCODING, Constants.ENCODING_GZIP);
085                        ServletOutputStream outputStream = theHttpResponse.getOutputStream();
086                        myWriter = new OutputStreamWriter(new GZIPOutputStream(outputStream), StandardCharsets.UTF_8);
087                        return myWriter;
088                }
089
090                myWriter = theHttpResponse.getWriter();
091                return myWriter;
092        }
093
094        private void addHeaders() {
095                HttpServletResponse httpResponse = getRequestDetails().getServletResponse();
096                getRequestDetails().getServer().addHeadersToResponse(httpResponse);
097                for (Entry<String, List<String>> header : getHeaders().entrySet()) {
098                        String key = header.getKey();
099                        key = sanitizeHeaderField(key);
100                        boolean first = true;
101                        for (String value : header.getValue()) {
102                                value = sanitizeHeaderField(value);
103
104                                // existing headers should be overridden
105                                if (first) {
106                                        httpResponse.setHeader(key, value);
107                                        first = false;
108                                } else {
109                                        httpResponse.addHeader(key, value);
110                                }
111                        }
112                }
113        }
114
115        @Override
116        public final Object commitResponse(@Nonnull Closeable theWriterOrOutputStream) {
117                IoUtil.closeQuietly(theWriterOrOutputStream);
118                return null;
119        }
120
121        static String sanitizeHeaderField(String theKey) {
122                return StringUtils.replaceChars(theKey, "\r\n", null);
123        }
124}