001/*-
002 * #%L
003 * HAPI FHIR - Client 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.client.tls;
021
022import ca.uhn.fhir.i18n.Msg;
023import ca.uhn.fhir.tls.BaseStoreInfo;
024import ca.uhn.fhir.tls.KeyStoreInfo;
025import ca.uhn.fhir.tls.PathType;
026import ca.uhn.fhir.tls.TlsAuthentication;
027import ca.uhn.fhir.tls.TrustStoreInfo;
028import jakarta.annotation.Nonnull;
029import org.apache.commons.lang3.Validate;
030import org.apache.http.conn.ssl.DefaultHostnameVerifier;
031import org.apache.http.conn.ssl.NoopHostnameVerifier;
032import org.apache.http.conn.ssl.TrustSelfSignedStrategy;
033import org.apache.http.ssl.PrivateKeyStrategy;
034import org.apache.http.ssl.SSLContextBuilder;
035import org.apache.http.ssl.SSLContexts;
036
037import java.io.FileInputStream;
038import java.io.InputStream;
039import java.security.KeyStore;
040import java.util.Optional;
041import javax.net.ssl.HostnameVerifier;
042import javax.net.ssl.SSLContext;
043import javax.net.ssl.TrustManager;
044import javax.net.ssl.TrustManagerFactory;
045import javax.net.ssl.X509TrustManager;
046
047import static org.apache.commons.lang3.StringUtils.isNotBlank;
048
049public class TlsAuthenticationSvc {
050
051        private TlsAuthenticationSvc() {}
052
053        public static SSLContext createSslContext(@Nonnull TlsAuthentication theTlsAuthentication) {
054                Validate.notNull(theTlsAuthentication, "theTlsAuthentication cannot be null");
055
056                try {
057                        SSLContextBuilder contextBuilder = SSLContexts.custom();
058
059                        if (theTlsAuthentication.getKeyStoreInfo().isPresent()) {
060                                KeyStoreInfo keyStoreInfo =
061                                                theTlsAuthentication.getKeyStoreInfo().get();
062                                PrivateKeyStrategy privateKeyStrategy = null;
063                                if (isNotBlank(keyStoreInfo.getAlias())) {
064                                        privateKeyStrategy = (aliases, socket) -> keyStoreInfo.getAlias();
065                                }
066                                KeyStore keyStore = createKeyStore(keyStoreInfo);
067                                contextBuilder.loadKeyMaterial(keyStore, keyStoreInfo.getKeyPass(), privateKeyStrategy);
068                        }
069
070                        if (theTlsAuthentication.getTrustStoreInfo().isPresent()) {
071                                TrustStoreInfo trustStoreInfo =
072                                                theTlsAuthentication.getTrustStoreInfo().get();
073                                KeyStore trustStore = createKeyStore(trustStoreInfo);
074                                contextBuilder.loadTrustMaterial(trustStore, TrustSelfSignedStrategy.INSTANCE);
075                        }
076
077                        return contextBuilder.build();
078                } catch (Exception e) {
079                        throw new TlsAuthenticationException(Msg.code(2102) + "Failed to create SSLContext", e);
080                }
081        }
082
083        public static KeyStore createKeyStore(BaseStoreInfo theStoreInfo) {
084                try {
085                        KeyStore keyStore = KeyStore.getInstance(theStoreInfo.getType().toString());
086
087                        if (PathType.RESOURCE.equals(theStoreInfo.getPathType())) {
088                                try (InputStream inputStream =
089                                                TlsAuthenticationSvc.class.getResourceAsStream(theStoreInfo.getFilePath())) {
090                                        validateKeyStoreExists(inputStream);
091                                        keyStore.load(inputStream, theStoreInfo.getStorePass());
092                                }
093                        } else if (PathType.FILE.equals(theStoreInfo.getPathType())) {
094                                try (InputStream inputStream = new FileInputStream(theStoreInfo.getFilePath())) {
095                                        validateKeyStoreExists(inputStream);
096                                        keyStore.load(inputStream, theStoreInfo.getStorePass());
097                                }
098                        }
099                        return keyStore;
100                } catch (Exception e) {
101                        throw new TlsAuthenticationException(Msg.code(2103) + "Failed to create KeyStore", e);
102                }
103        }
104
105        public static void validateKeyStoreExists(InputStream theInputStream) {
106                if (theInputStream == null) {
107                        throw new TlsAuthenticationException(Msg.code(2116) + "Keystore does not exists");
108                }
109        }
110
111        public static X509TrustManager createTrustManager(Optional<TrustStoreInfo> theTrustStoreInfo) {
112                try {
113                        TrustManagerFactory trustManagerFactory =
114                                        TrustManagerFactory.getInstance(TrustManagerFactory.getDefaultAlgorithm());
115                        if (!theTrustStoreInfo.isPresent()) {
116                                trustManagerFactory.init((KeyStore) null); // Load Trust Manager Factory with default Java truststore
117                        } else {
118                                TrustStoreInfo trustStoreInfo = theTrustStoreInfo.get();
119                                KeyStore trustStore = createKeyStore(trustStoreInfo);
120                                trustManagerFactory.init(trustStore);
121                        }
122                        for (TrustManager trustManager : trustManagerFactory.getTrustManagers()) {
123                                if (trustManager instanceof X509TrustManager) {
124                                        return (X509TrustManager) trustManager;
125                                }
126                        }
127                        throw new TlsAuthenticationException(Msg.code(2104) + "Could not find X509TrustManager");
128                } catch (Exception e) {
129                        throw new TlsAuthenticationException(Msg.code(2105) + "Failed to create X509TrustManager");
130                }
131        }
132
133        public static HostnameVerifier createHostnameVerifier(Optional<TrustStoreInfo> theTrustStoreInfo) {
134                return theTrustStoreInfo.isPresent() ? new DefaultHostnameVerifier() : new NoopHostnameVerifier();
135        }
136
137        public static class TlsAuthenticationException extends RuntimeException {
138                private static final long serialVersionUID = 1l;
139
140                public TlsAuthenticationException(String theMessage, Throwable theCause) {
141                        super(theMessage, theCause);
142                }
143
144                public TlsAuthenticationException(String theMessage) {
145                        super(theMessage);
146                }
147        }
148}