001/*-
002 * #%L
003 * Smile CDR - CDR
004 * %%
005 * Copyright (C) 2016 - 2025 Smile CDR, Inc.
006 * %%
007 * All rights reserved.
008 * #L%
009 */
010package ca.cdr.api.util;
011
012import jakarta.annotation.Nullable;
013import jakarta.servlet.http.HttpServletRequest;
014import org.slf4j.Logger;
015import org.slf4j.LoggerFactory;
016import org.springframework.http.HttpHeaders;
017import org.springframework.web.util.ForwardedHeaderUtils;
018
019import java.net.InetSocketAddress;
020import java.net.URI;
021import java.net.URISyntaxException;
022import java.util.Enumeration;
023
024import static org.apache.commons.lang3.StringUtils.isBlank;
025import static org.apache.commons.lang3.StringUtils.isNotBlank;
026
027public class HttpRequestUtil {
028        private static final Logger ourLog = LoggerFactory.getLogger(HttpRequestUtil.class);
029
030        /**
031         * Spec header. see:
032         * <a href="https://datatracker.ietf.org/doc/html/rfc7239">Forwarded spec</a>
033         */
034        private HttpRequestUtil() {}
035
036        /**
037         * Unknown identifier
038         */
039        private static final String UNKNOWN = "unknown";
040
041        /**
042         * Gets the best attempt at the originating
043         * client ip address from the incoming HttpServletRequest.
044         *
045         * NB: Because proxies can remove or change ip addresses (depending on
046         * implementation), this cannot be <b>guaranteed</b> to be the originating
047         * client ip address. But if the request is made directly, or proxies
048         * are handling them in standardized ways, its most likely going to be.
049         *
050         * In order of priority, this method will prioritize:
051         * * Forwarded header as defined in <a href="https://datatracker.ietf.org/doc/html/rfc7239">Forwarded spec</a>
052         * * X-Forwarded-for header (a standard, but not a spec)
053         * * the address on the request itself
054         *
055         * @param theRequest - the request
056         * @return the ip address of the client or first recorded ip address.
057         */
058        public static String getRemoteAddressOfServletRequest(HttpServletRequest theRequest) {
059                URI uri;
060                try {
061                        uri = new URI(theRequest.getRequestURL().toString());
062                } catch (URISyntaxException ex) {
063                        ourLog.warn(
064                                        "Unable to acquire remote address from request. This can affect log tracing, but should not affect functionality.",
065                                        ex);
066                        return null;
067                }
068                HttpHeaders headers = new HttpHeaders();
069                Enumeration<String> headerNames = theRequest.getHeaderNames();
070                while (headerNames.hasMoreElements()) {
071                        String h = headerNames.nextElement();
072                        Enumeration<String> values = theRequest.getHeaders(h);
073                        while (values.hasMoreElements()) {
074                                headers.add(h, values.nextElement());
075                        }
076                }
077                return getRemoteAddressFromHeaders(uri, headers, theRequest.getRemoteAddr());
078        }
079
080        public static String getRemoteAddressFromHeaders(URI theUri, HttpHeaders theHeaders, @Nullable String theDefault) {
081                InetSocketAddress addr = ForwardedHeaderUtils.parseForwardedFor(theUri, theHeaders, null);
082                String originIp = null;
083                if (addr != null && isNotBlank(addr.getHostName())) {
084                        originIp = addr.getHostName();
085                }
086
087                if (isBlank(originIp)) {
088                        originIp = theDefault;
089                }
090
091                if (isBlank(originIp)) {
092                        originIp = UNKNOWN;
093                }
094
095                return originIp;
096        }
097}