001/*-
002 * #%L
003 * HAPI FHIR - Core Library
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.util;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.rest.api.EncodingEnum;
025import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
026import com.google.common.base.Charsets;
027import jakarta.annotation.Nonnull;
028import org.apache.commons.io.IOUtils;
029import org.apache.commons.io.input.BOMInputStream;
030import org.hl7.fhir.instance.model.api.IBaseResource;
031import org.slf4j.Logger;
032import org.slf4j.LoggerFactory;
033
034import java.io.IOException;
035import java.io.InputStream;
036import java.io.InputStreamReader;
037import java.io.Reader;
038import java.nio.charset.StandardCharsets;
039import java.util.function.Function;
040import java.util.zip.GZIPInputStream;
041
042/**
043 * Use this API with caution, it may change!
044 */
045public class ClasspathUtil {
046
047        private static final Logger ourLog = LoggerFactory.getLogger(ClasspathUtil.class);
048
049        /**
050         * Non instantiable
051         */
052        private ClasspathUtil() {
053                // nothing
054        }
055
056        /**
057         * Load a classpath resource, throw an {@link InternalErrorException} if not found
058         *
059         * @throws InternalErrorException If the resource can't be found
060         */
061        public static String loadResource(String theClasspath) throws InternalErrorException {
062                return loadResource(theClasspath, Function.identity());
063        }
064
065        /**
066         * Load a classpath resource, throw an {@link InternalErrorException} if not found
067         *
068         * @throws InternalErrorException If the resource can't be found
069         */
070        @Nonnull
071        public static InputStream loadResourceAsStream(String theClasspath) throws InternalErrorException {
072                String classpath = theClasspath;
073                if (classpath.startsWith("classpath:")) {
074                        classpath = classpath.substring("classpath:".length());
075                }
076
077                InputStream retVal = ClasspathUtil.class.getResourceAsStream(classpath);
078                if (retVal == null) {
079                        if (classpath.startsWith("/")) {
080                                retVal = ClasspathUtil.class.getResourceAsStream(classpath.substring(1));
081                        } else {
082                                retVal = ClasspathUtil.class.getResourceAsStream("/" + classpath);
083                        }
084                        if (retVal == null) {
085                                throw new InternalErrorException(Msg.code(1758) + "Unable to find classpath resource: " + classpath);
086                        }
087                }
088                return retVal;
089        }
090
091        public static Reader loadResourceAsReader(String theClasspath) {
092                return new InputStreamReader(loadResourceAsStream(theClasspath), StandardCharsets.UTF_8);
093        }
094
095        /**
096         * Load a classpath resource, throw an {@link InternalErrorException} if not found
097         */
098        @Nonnull
099        public static String loadResource(String theClasspath, Function<InputStream, InputStream> theStreamTransform) {
100                try (InputStream stream = loadResourceAsStream(theClasspath)) {
101                        InputStream newStream = theStreamTransform.apply(stream);
102                        return IOUtils.toString(newStream, Charsets.UTF_8);
103                } catch (IOException e) {
104                        throw new InternalErrorException(Msg.code(1759) + e);
105                }
106        }
107
108        @Nonnull
109        public static String loadCompressedResource(String theClasspath) {
110                Function<InputStream, InputStream> streamTransform = t -> {
111                        try {
112                                return new GZIPInputStream(t);
113                        } catch (IOException e) {
114                                throw new InternalErrorException(Msg.code(1760) + e);
115                        }
116                };
117                return loadResource(theClasspath, streamTransform);
118        }
119
120        /**
121         * Load a classpath resource, throw an {@link InternalErrorException} if not found
122         *
123         * @since 6.4.0
124         */
125        @Nonnull
126        public static <T extends IBaseResource> T loadCompressedResource(
127                        FhirContext theCtx, Class<T> theType, String theClasspath) {
128                String resource = loadCompressedResource(theClasspath);
129                return parseResource(theCtx, theType, resource);
130        }
131
132        @Nonnull
133        public static <T extends IBaseResource> T loadResource(FhirContext theCtx, Class<T> theType, String theClasspath) {
134                String raw = loadResource(theClasspath);
135                return parseResource(theCtx, theType, raw);
136        }
137
138        private static <T extends IBaseResource> T parseResource(FhirContext theCtx, Class<T> theType, String raw) {
139                return EncodingEnum.detectEncodingNoDefault(raw).newParser(theCtx).parseResource(theType, raw);
140        }
141
142        public static void close(InputStream theInput) {
143                try {
144                        if (theInput != null) {
145                                theInput.close();
146                        }
147                } catch (IOException e) {
148                        ourLog.debug("Closing InputStream threw exception", e);
149                }
150        }
151
152        public static Function<InputStream, InputStream> withBom() {
153                return t -> new BOMInputStream(t);
154        }
155
156        public static byte[] loadResourceAsByteArray(String theClasspath) {
157                InputStream stream = loadResourceAsStream(theClasspath);
158                try {
159                        return IOUtils.toByteArray(stream);
160                } catch (IOException e) {
161                        throw new InternalErrorException(Msg.code(1761) + e);
162                } finally {
163                        close(stream);
164                }
165        }
166}