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;
021
022import ca.uhn.fhir.rest.api.Constants;
023import jakarta.annotation.Nullable;
024import jakarta.servlet.ServletRequest;
025import jakarta.servlet.http.HttpServletRequest;
026import org.apache.commons.lang3.RandomStringUtils;
027import org.slf4j.Logger;
028import org.slf4j.LoggerFactory;
029
030import static org.apache.commons.lang3.StringUtils.isBlank;
031import static org.apache.commons.lang3.StringUtils.isNotBlank;
032
033public class ServletRequestTracing {
034        private static final Logger ourLog = LoggerFactory.getLogger(ServletRequestTracing.class);
035        public static final String ATTRIBUTE_REQUEST_ID =
036                        ServletRequestTracing.class.getName() + '.' + Constants.HEADER_REQUEST_ID;
037
038        ServletRequestTracing() {}
039
040        /**
041         * Assign a tracing id to this request, using
042         * the X-Request-ID if present and compatible.
043         *
044         * If none present, generate a 64 random alpha-numeric string that is not
045         * cryptographically secure.
046         *
047         * @param theServletRequest the request to trace
048         * @return the tracing id
049         */
050        public static String getOrGenerateRequestId(ServletRequest theServletRequest) {
051                String requestId = maybeGetRequestId(theServletRequest);
052                if (isBlank(requestId)) {
053                        requestId = RandomStringUtils.randomAlphanumeric(Constants.REQUEST_ID_LENGTH);
054                }
055
056                ourLog.debug("Assigned tracing id {}", requestId);
057
058                theServletRequest.setAttribute(ATTRIBUTE_REQUEST_ID, requestId);
059
060                return requestId;
061        }
062
063        @Nullable
064        public static String maybeGetRequestId(ServletRequest theServletRequest) {
065                // have we already seen this request?
066                String requestId = (String) theServletRequest.getAttribute(ATTRIBUTE_REQUEST_ID);
067
068                if (requestId == null && theServletRequest instanceof HttpServletRequest) {
069                        // Also applies to non-FHIR (e.g. admin-json) requests).
070                        HttpServletRequest request = (HttpServletRequest) theServletRequest;
071                        requestId = request.getHeader(Constants.HEADER_REQUEST_ID);
072                        if (isNotBlank(requestId)) {
073                                for (char nextChar : requestId.toCharArray()) {
074                                        if (!Character.isLetterOrDigit(nextChar)) {
075                                                if (nextChar != '.' && nextChar != '-' && nextChar != '_' && nextChar != ' ') {
076                                                        requestId = null;
077                                                        break;
078                                                }
079                                        }
080                                }
081                        }
082                }
083                return requestId;
084        }
085}