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.interceptor; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.i18n.Msg; 024import ca.uhn.fhir.interceptor.api.Hook; 025import ca.uhn.fhir.interceptor.api.Interceptor; 026import ca.uhn.fhir.interceptor.api.Pointcut; 027import ca.uhn.fhir.rest.api.Constants; 028import ca.uhn.fhir.rest.api.RestOperationTypeEnum; 029import ca.uhn.fhir.rest.api.server.RequestDetails; 030import ca.uhn.fhir.rest.server.RestfulServer; 031import ca.uhn.fhir.rest.server.RestfulServerUtils; 032import ca.uhn.fhir.rest.server.exceptions.AuthenticationException; 033import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 034import jakarta.servlet.http.HttpServletRequest; 035import jakarta.servlet.http.HttpServletResponse; 036import org.hl7.fhir.instance.model.api.IBaseResource; 037import org.hl7.fhir.instance.model.api.IPrimitiveType; 038 039import java.io.IOException; 040import java.util.Collections; 041import java.util.HashSet; 042import java.util.Optional; 043import java.util.Set; 044 045import static org.apache.commons.lang3.StringUtils.isBlank; 046 047/** 048 * This interceptor allows a client to request that a Media resource be 049 * served as the raw contents of the resource, assuming either: 050 * <ul> 051 * <li>The client explicitly requests the correct content type using the Accept header</li> 052 * <li>The client explicitly requests raw output by adding the parameter <code>_output=data</code></li> 053 * </ul> 054 */ 055@Interceptor 056public class ServeMediaResourceRawInterceptor { 057 058 public static final String MEDIA_CONTENT_CONTENT_TYPE_OPT = "Media.content.contentType"; 059 060 private static final Set<RestOperationTypeEnum> RESPOND_TO_OPERATION_TYPES; 061 062 static { 063 Set<RestOperationTypeEnum> respondToOperationTypes = new HashSet<>(); 064 respondToOperationTypes.add(RestOperationTypeEnum.READ); 065 respondToOperationTypes.add(RestOperationTypeEnum.VREAD); 066 RESPOND_TO_OPERATION_TYPES = Collections.unmodifiableSet(respondToOperationTypes); 067 } 068 069 @Hook(value = Pointcut.SERVER_OUTGOING_RESPONSE, order = InterceptorOrders.SERVE_MEDIA_RESOURCE_RAW_INTERCEPTOR) 070 public boolean outgoingResponse( 071 RequestDetails theRequestDetails, 072 IBaseResource theResponseObject, 073 HttpServletRequest theServletRequest, 074 HttpServletResponse theServletResponse) 075 throws AuthenticationException { 076 if (theResponseObject == null) { 077 return true; 078 } 079 080 FhirContext context = theRequestDetails.getFhirContext(); 081 String resourceName = context.getResourceType(theResponseObject); 082 083 // Are we serving a FHIR read request on the Media resource type 084 if (!"Media".equals(resourceName) 085 || !RESPOND_TO_OPERATION_TYPES.contains(theRequestDetails.getRestOperationType())) { 086 return true; 087 } 088 089 // What is the content type of the Media resource we're returning? 090 String contentType = null; 091 Optional<IPrimitiveType> contentTypeOpt = context.newFluentPath() 092 .evaluateFirst(theResponseObject, MEDIA_CONTENT_CONTENT_TYPE_OPT, IPrimitiveType.class); 093 if (contentTypeOpt.isPresent()) { 094 contentType = contentTypeOpt.get().getValueAsString(); 095 } 096 097 // What is the data of the Media resource we're returning? 098 IPrimitiveType<byte[]> data = null; 099 Optional<IPrimitiveType> dataOpt = 100 context.newFluentPath().evaluateFirst(theResponseObject, "Media.content.data", IPrimitiveType.class); 101 if (dataOpt.isPresent()) { 102 data = dataOpt.get(); 103 } 104 105 if (isBlank(contentType) || data == null) { 106 return true; 107 } 108 109 RestfulServerUtils.ResponseEncoding responseEncoding = 110 RestfulServerUtils.determineResponseEncodingNoDefault(theRequestDetails, null, contentType); 111 if (responseEncoding != null) { 112 if (contentType.equals(responseEncoding.getContentType())) { 113 returnRawResponse(theRequestDetails, theServletResponse, contentType, data); 114 return false; 115 } 116 } 117 118 String[] outputParam = theRequestDetails.getParameters().get("_output"); 119 if (outputParam != null && "data".equals(outputParam[0])) { 120 returnRawResponse(theRequestDetails, theServletResponse, contentType, data); 121 return false; 122 } 123 124 return true; 125 } 126 127 private void returnRawResponse( 128 RequestDetails theRequestDetails, 129 HttpServletResponse theServletResponse, 130 String theContentType, 131 IPrimitiveType<byte[]> theData) { 132 theServletResponse.setStatus(200); 133 if (theRequestDetails.getServer() instanceof RestfulServer) { 134 RestfulServer rs = (RestfulServer) theRequestDetails.getServer(); 135 rs.addHeadersToResponse(theServletResponse); 136 } 137 138 theServletResponse.addHeader(Constants.HEADER_CONTENT_TYPE, theContentType); 139 140 // Write the response 141 try { 142 theServletResponse.getOutputStream().write(theData.getValue()); 143 theServletResponse.getOutputStream().close(); 144 } catch (IOException e) { 145 throw new InternalErrorException(Msg.code(321) + e); 146 } 147 } 148}