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.parser.DataFormatException; 028import ca.uhn.fhir.rest.api.Constants; 029import ca.uhn.fhir.rest.api.SummaryEnum; 030import ca.uhn.fhir.rest.api.server.IRestfulResponse; 031import ca.uhn.fhir.rest.api.server.RequestDetails; 032import ca.uhn.fhir.rest.server.RestfulServerUtils; 033import ca.uhn.fhir.rest.server.exceptions.BaseServerResponseException; 034import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 035import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 036import ca.uhn.fhir.rest.server.exceptions.UnclassifiedServerFailureException; 037import ca.uhn.fhir.rest.server.method.BaseResourceReturningMethodBinding; 038import ca.uhn.fhir.rest.server.servlet.ServletRestfulResponse; 039import ca.uhn.fhir.util.OperationOutcomeUtil; 040import jakarta.servlet.ServletException; 041import jakarta.servlet.http.HttpServletRequest; 042import jakarta.servlet.http.HttpServletResponse; 043import org.apache.commons.collections4.map.HashedMap; 044import org.apache.commons.lang3.exception.ExceptionUtils; 045import org.hl7.fhir.instance.model.api.IBaseOperationOutcome; 046 047import java.io.IOException; 048import java.util.Collection; 049import java.util.Collections; 050import java.util.List; 051import java.util.Map; 052import java.util.Map.Entry; 053import java.util.Set; 054 055import static org.apache.commons.lang3.StringUtils.isNotBlank; 056import static org.apache.http.HttpHeaders.CONTENT_ENCODING; 057 058@Interceptor 059public class ExceptionHandlingInterceptor { 060 061 public static final String PROCESSING = Constants.OO_INFOSTATUS_PROCESSING; 062 private static final org.slf4j.Logger ourLog = 063 org.slf4j.LoggerFactory.getLogger(ExceptionHandlingInterceptor.class); 064 public static final Set<SummaryEnum> SUMMARY_MODE = Collections.singleton(SummaryEnum.FALSE); 065 private Class<?>[] myReturnStackTracesForExceptionTypes; 066 067 @Hook(Pointcut.SERVER_HANDLE_EXCEPTION) 068 public boolean handleException( 069 RequestDetails theRequestDetails, 070 BaseServerResponseException theException, 071 HttpServletRequest theRequest, 072 HttpServletResponse theResponse) 073 throws ServletException, IOException { 074 handleException(theRequestDetails, theException); 075 return false; 076 } 077 078 public Object handleException(RequestDetails theRequestDetails, BaseServerResponseException theException) 079 throws ServletException, IOException { 080 IRestfulResponse response = theRequestDetails.getResponse(); 081 082 FhirContext ctx = theRequestDetails.getServer().getFhirContext(); 083 084 IBaseOperationOutcome oo = theException.getOperationOutcome(); 085 if (oo == null) { 086 oo = createOperationOutcome(theException, ctx); 087 } 088 089 int statusCode = theException.getStatusCode(); 090 091 // Add headers associated with the specific error code 092 if (theException.hasResponseHeaders()) { 093 Map<String, List<String>> additional = theException.getResponseHeaders(); 094 for (Entry<String, List<String>> next : additional.entrySet()) { 095 if (isNotBlank(next.getKey()) && next.getValue() != null) { 096 String nextKey = next.getKey(); 097 for (String nextValue : next.getValue()) { 098 response.addHeader(nextKey, nextValue); 099 } 100 } 101 } 102 } 103 104 String statusMessage = null; 105 if (theException instanceof UnclassifiedServerFailureException) { 106 String sm = theException.getMessage(); 107 if (isNotBlank(sm) && sm.indexOf('\n') == -1) { 108 statusMessage = sm; 109 } 110 } 111 112 BaseResourceReturningMethodBinding.callOutgoingFailureOperationOutcomeHook(theRequestDetails, oo); 113 try { 114 resetOutputStreamIfPossible(response); 115 } catch (Throwable t) { 116 ourLog.error( 117 "HAPI-FHIR was unable to reset the output stream during exception handling. The root causes follows:", 118 t); 119 } 120 121 return RestfulServerUtils.streamResponseAsResource( 122 theRequestDetails.getServer(), 123 oo, 124 SUMMARY_MODE, 125 statusCode, 126 false, 127 false, 128 theRequestDetails, 129 null, 130 null); 131 } 132 133 /** 134 * In some edge cases, the output stream is opened already by the point at which an exception is thrown. 135 * This is a failsafe to that the output stream is writeable in that case. This operation retains status code and headers, but clears the buffer. 136 * Also, it strips the content-encoding header if present, as the method outcome will negotiate its own. 137 */ 138 private void resetOutputStreamIfPossible(IRestfulResponse response) { 139 if (response.getClass().isAssignableFrom(ServletRestfulResponse.class)) { 140 ServletRestfulResponse servletRestfulResponse = (ServletRestfulResponse) response; 141 HttpServletResponse servletResponse = 142 servletRestfulResponse.getRequestDetails().getServletResponse(); 143 Collection<String> headerNames = servletResponse.getHeaderNames(); 144 Map<String, Collection<String>> oldHeaders = new HashedMap<>(); 145 for (String headerName : headerNames) { 146 oldHeaders.put(headerName, servletResponse.getHeaders(headerName)); 147 } 148 149 servletResponse.reset(); 150 oldHeaders.entrySet().stream() 151 .filter(entry -> !entry.getKey().equals(CONTENT_ENCODING)) 152 .forEach(entry -> { 153 entry.getValue().stream().forEach(value -> { 154 servletResponse.addHeader(entry.getKey(), value); 155 }); 156 }); 157 } 158 } 159 160 @Hook(Pointcut.SERVER_PRE_PROCESS_OUTGOING_EXCEPTION) 161 public BaseServerResponseException preProcessOutgoingException( 162 RequestDetails theRequestDetails, Throwable theException, HttpServletRequest theServletRequest) 163 throws ServletException { 164 BaseServerResponseException retVal; 165 if (theException instanceof DataFormatException) { 166 // Wrapping the DataFormatException as an InvalidRequestException so that it gets sent back to the client as 167 // a 400 response. 168 retVal = new InvalidRequestException(theException); 169 } else if (!(theException instanceof BaseServerResponseException)) { 170 retVal = new InternalErrorException(theException); 171 } else { 172 retVal = (BaseServerResponseException) theException; 173 } 174 175 if (retVal.getOperationOutcome() == null) { 176 retVal.setOperationOutcome(createOperationOutcome( 177 theException, theRequestDetails.getServer().getFhirContext())); 178 } 179 180 return retVal; 181 } 182 183 private IBaseOperationOutcome createOperationOutcome(Throwable theException, FhirContext ctx) 184 throws ServletException { 185 IBaseOperationOutcome oo = null; 186 if (theException instanceof BaseServerResponseException) { 187 oo = ((BaseServerResponseException) theException).getOperationOutcome(); 188 } 189 190 /* 191 * Generate an OperationOutcome to return, unless the exception throw by the resource provider had one 192 */ 193 if (oo == null) { 194 try { 195 oo = OperationOutcomeUtil.newInstance(ctx); 196 197 if (theException instanceof InternalErrorException) { 198 ourLog.error("Failure during REST processing", theException); 199 populateDetails(ctx, theException, oo); 200 } else if (theException instanceof BaseServerResponseException) { 201 int statusCode = ((BaseServerResponseException) theException).getStatusCode(); 202 203 // No stack traces for non-server internal errors 204 if (statusCode < 500) { 205 ourLog.warn("Failure during REST processing: {}", theException.toString()); 206 } else { 207 ourLog.warn("Failure during REST processing", theException); 208 } 209 210 BaseServerResponseException baseServerResponseException = 211 (BaseServerResponseException) theException; 212 populateDetails(ctx, theException, oo); 213 if (baseServerResponseException.getAdditionalMessages() != null) { 214 for (String next : baseServerResponseException.getAdditionalMessages()) { 215 OperationOutcomeUtil.addIssue(ctx, oo, "error", next, null, PROCESSING); 216 } 217 } 218 } else { 219 ourLog.error("Failure during REST processing: " + theException.toString(), theException); 220 populateDetails(ctx, theException, oo); 221 } 222 } catch (Exception e1) { 223 ourLog.error("Failed to instantiate OperationOutcome resource instance", e1); 224 throw new ServletException( 225 Msg.code(328) + "Failed to instantiate OperationOutcome resource instance", e1); 226 } 227 } else { 228 ourLog.error("Unknown error during processing", theException); 229 } 230 return oo; 231 } 232 233 private void populateDetails(FhirContext theCtx, Throwable theException, IBaseOperationOutcome theOo) { 234 if (myReturnStackTracesForExceptionTypes != null) { 235 for (Class<?> next : myReturnStackTracesForExceptionTypes) { 236 if (next.isAssignableFrom(theException.getClass())) { 237 String detailsValue = 238 theException.getMessage() + "\n\n" + ExceptionUtils.getStackTrace(theException); 239 OperationOutcomeUtil.addIssue(theCtx, theOo, "error", detailsValue, null, PROCESSING); 240 return; 241 } 242 } 243 } 244 245 OperationOutcomeUtil.addIssue(theCtx, theOo, "error", theException.getMessage(), null, PROCESSING); 246 } 247 248 /** 249 * If any server methods throw an exception which extends any of the given exception types, the exception stack trace will be returned to the user. This can be useful for helping to diagnose 250 * issues, but may not be desirable for production situations. 251 * 252 * @param theExceptionTypes 253 * The exception types for which to return the stack trace to the user. 254 * @return Returns an instance of this interceptor, to allow for easy method chaining. 255 */ 256 public ExceptionHandlingInterceptor setReturnStackTracesForExceptionTypes(Class<?>... theExceptionTypes) { 257 myReturnStackTracesForExceptionTypes = theExceptionTypes; 258 return this; 259 } 260}