All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.helidon.config.ClasspathSourceHelper Maven / Gradle / Ivy

There is a newer version: 4.1.1
Show newest version
/*
 * Copyright (c) 2017, 2023 Oracle and/or its affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package io.helidon.config;

import java.io.InputStream;
import java.lang.System.Logger.Level;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.Instant;
import java.util.function.BiFunction;

/**
 * Utilities for file-related source classes.
 *
 * @see io.helidon.config.ClasspathConfigSource
 * @see ClasspathOverrideSource
 */
class ClasspathSourceHelper {

    private static final System.Logger LOGGER = System.getLogger(ClasspathSourceHelper.class.getName());

    private ClasspathSourceHelper() {
        throw new AssertionError("Instantiation not allowed.");
    }

    static String uid(String resourceName) {
        try {
            Path resourcePath = ClasspathSourceHelper.resourcePath(resourceName);
            if (resourcePath != null) {
                //Backwards slash replacement is workaround used because of windows compatibility.
                return resourcePath.toString().replace('\\', '/');
            }
        } catch (Exception ex) {
            //ignore it
            LOGGER.log(Level.DEBUG,
                       "Not possible to get filesystem path for resource '" + resourceName
                               + "'. Resource's name is used as ConfigSource URI.",
                       ex);
        }
        return resourceName;
    }

    static Path resourcePath(String resourceName) throws URISyntaxException {
        URL resourceUrl = Thread.currentThread().getContextClassLoader().getResource(resourceName);
        if (resourceUrl != null) {
            // this can only work if we are using a file based classloader (which may not be always the case)
            // we may load classes from http, or from specific loaders, such as in Graal native image
            URI uri = resourceUrl.toURI();
            if ("file".equals(uri.getScheme())) {
                return Paths.get(uri);
            } else {
                return null;
            }
        } else {
            return null;
        }
    }

    static Instant resourceTimestamp(String resourceName) {
        try {
            Path resourcePath = resourcePath(resourceName);
            if (resourcePath != null) {
                return Files.getLastModifiedTime(resourcePath).toInstant();
            }
        } catch (Exception ex) {
            LOGGER.log(Level.DEBUG, "Error to get resource '" + resourceName + "' last modified time.", ex);
        }
        return Instant.EPOCH;
    }

    static  T content(String resource,
                         String description,
                         BiFunction processor) throws ConfigException {
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();

        InputStream inputStream = classLoader.getResourceAsStream(resource);

        if (inputStream == null) {
            LOGGER.log(Level.DEBUG,
                       String.format("Error to get %s using %s CONTEXT ClassLoader.", description, classLoader));
            throw new ConfigException(description + " does not exist. Used ClassLoader: " + classLoader);
        }
        Instant resourceTimestamp = resourceTimestamp(resource);
        try {
            LOGGER.log(Level.DEBUG,
                       String.format("Getting content from '%s'. Last modified at %s. Used ClassLoader: %s",
                                     resourcePath(resource), resourceTimestamp, classLoader));
        } catch (Exception ex) {
            LOGGER.log(Level.DEBUG, "Error to get resource '" + resource + "' path. Used ClassLoader: " + classLoader, ex);
        }

        return processor.apply(inputStream, resourceTimestamp);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy