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.jar;
021
022import ca.uhn.fhir.util.XmlUtil;
023
024import java.io.InputStream;
025import java.net.MalformedURLException;
026import java.net.URL;
027import java.util.jar.Attributes;
028import java.util.jar.Manifest;
029
030public class DependencyLogImpl implements IDependencyLog {
031        private static final Attributes.Name BUNDLE_SYMBOLIC_NAME = new Attributes.Name("Bundle-SymbolicName");
032        private static final Attributes.Name BUNDLE_VENDOR = new Attributes.Name("Bundle-Vendor");
033        private static final Attributes.Name BUNDLE_VERSION = new Attributes.Name("Bundle-Version");
034        private static final Attributes.Name IMPLEMENTATION_TITLE = new Attributes.Name("Implementation-Title");
035        private static final Attributes.Name IMPLEMENTATION_VENDOR = new Attributes.Name("Implementation-Vendor");
036        private static final Attributes.Name IMPLEMENTATION_VERSION = new Attributes.Name("Implementation-Version");
037        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(XmlUtil.class);
038
039        @Override
040        public void logStaxImplementation(Class<?> theClass) {
041                try {
042                        URL rootUrl = getRootUrlForClass(theClass);
043                        if (rootUrl == null) {
044                                ourLog.info("Unable to determine location of StAX implementation containing class");
045                        } else {
046                                Manifest manifest;
047                                URL metaInfUrl = new URL(rootUrl, "META-INF/MANIFEST.MF");
048                                InputStream is = metaInfUrl.openStream();
049                                try {
050                                        manifest = new Manifest(is);
051                                } finally {
052                                        is.close();
053                                }
054                                Attributes attrs = manifest.getMainAttributes();
055                                String title = attrs.getValue(IMPLEMENTATION_TITLE);
056                                String symbolicName = attrs.getValue(BUNDLE_SYMBOLIC_NAME);
057                                if (symbolicName != null) {
058                                        int i = symbolicName.indexOf(';');
059                                        if (i != -1) {
060                                                symbolicName = symbolicName.substring(0, i);
061                                        }
062                                }
063                                String vendor = attrs.getValue(IMPLEMENTATION_VENDOR);
064                                if (vendor == null) {
065                                        vendor = attrs.getValue(BUNDLE_VENDOR);
066                                }
067                                String version = attrs.getValue(IMPLEMENTATION_VERSION);
068                                if (version == null) {
069                                        version = attrs.getValue(BUNDLE_VERSION);
070                                }
071                                if (ourLog.isDebugEnabled()) {
072                                        ourLog.debug(
073                                                        "FHIR XML procesing will use StAX implementation at {}\n  Title:         {}\n  Symbolic name: {}\n  Vendor:        {}\n  Version:       {}",
074                                                        new Object[] {rootUrl, title, symbolicName, vendor, version});
075                                } else {
076                                        ourLog.info("FHIR XML procesing will use StAX implementation '{}' version '{}'", title, version);
077                                }
078                        }
079                } catch (Throwable e) {
080                        ourLog.info("Unable to determine StAX implementation: " + e.getMessage());
081                }
082        }
083
084        private static URL getRootUrlForClass(Class<?> cls) {
085                ClassLoader classLoader = cls.getClassLoader();
086                String resource = cls.getName().replace('.', '/') + ".class";
087                if (classLoader == null) {
088                        // A null class loader means the bootstrap class loader. In this case we use the
089                        // system class loader. This is safe since we can assume that the system class
090                        // loader uses parent first as delegation policy.
091                        classLoader = ClassLoader.getSystemClassLoader();
092                }
093                URL url = classLoader.getResource(resource);
094                if (url == null) {
095                        return null;
096                }
097                String file = url.getFile();
098                if (file.endsWith(resource)) {
099                        try {
100                                return new URL(
101                                                url.getProtocol(),
102                                                url.getHost(),
103                                                url.getPort(),
104                                                file.substring(0, file.length() - resource.length()));
105                        } catch (MalformedURLException ex) {
106                                return null;
107                        }
108                }
109                return null;
110        }
111}