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

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

There is a newer version: 4.1.1
Show newest version
/*
 * Copyright (c) 2017, 2022 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.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Optional;
import java.util.function.Function;

import io.helidon.common.media.type.MediaType;
import io.helidon.common.media.type.MediaTypes;
import io.helidon.config.FileSourceHelper.DataAndDigest;
import io.helidon.config.spi.ChangeWatcher;
import io.helidon.config.spi.ConfigParser;
import io.helidon.config.spi.ConfigSource;
import io.helidon.config.spi.ParsableSource;
import io.helidon.config.spi.PollableSource;
import io.helidon.config.spi.PollingStrategy;
import io.helidon.config.spi.WatchableSource;

/**
 * {@link ConfigSource} implementation that loads configuration content from a file on a filesystem.
 *
 * @see io.helidon.config.FileConfigSource.Builder
 */
public class FileConfigSource extends AbstractConfigSource
        implements WatchableSource, ParsableSource, PollableSource {

    private static final System.Logger LOGGER = System.getLogger(FileConfigSource.class.getName());
    private static final String PATH_KEY = "path";

    private final Path filePath;

    /**
     * Create a new file config source.
     *
     * @param builder builder with configured path and other options of this source
     */
    FileConfigSource(Builder builder) {
        super(builder);

        this.filePath = builder.path;
    }

    /**
     * Initializes config source instance from configuration properties.
     * 

* Mandatory {@code properties}, see {@link io.helidon.config.ConfigSources#file(String)}: *

    *
  • {@code path} - type {@link Path}
  • *
* Optional {@code properties}: see {@link AbstractConfigSourceBuilder#config(Config)}. * * @param metaConfig meta-configuration used to initialize returned config source instance from. * @return new instance of config source described by {@code metaConfig} * @throws MissingValueException in case the configuration tree does not contain all expected sub-nodes * required by the mapper implementation to provide instance of Java type. * @throws ConfigMappingException in case the mapper fails to map the (existing) configuration tree represented by the * supplied configuration node to an instance of a given Java type. * @see io.helidon.config.ConfigSources#file(String) * @see AbstractConfigSourceBuilder#config(Config) */ public static FileConfigSource create(Config metaConfig) throws ConfigMappingException, MissingValueException { return FileConfigSource.builder() .config(metaConfig) .build(); } /** * Get a builder instance to create a new config source. * @return a fluent API builder */ public static Builder builder() { return new Builder(); } @Override protected String uid() { return filePath.toString(); } @Override public boolean isModified(byte[] stamp) { return FileSourceHelper.isModified(filePath, stamp); } @Override public Path target() { return filePath; } @Override public Optional load() throws ConfigException { LOGGER.log(System.Logger.Level.TRACE, String.format("Getting content from '%s'", filePath)); // now we need to create all the necessary steps in one go, to make sure the digest matches the file Optional dataAndDigest = FileSourceHelper.readDataAndDigest(filePath); if (dataAndDigest.isEmpty()) { return Optional.empty(); } DataAndDigest dad = dataAndDigest.get(); InputStream dataStream = new ByteArrayInputStream(dad.data()); /* * Build the content */ var builder = ConfigParser.Content.builder() .stamp(dad.digest()) .data(dataStream); MediaTypes.detectType(filePath).ifPresent(builder::mediaType); return Optional.of(builder.build()); } @Override public Function> relativeResolver() { return it -> { Path path = filePath.getParent().resolve(it); if (Files.exists(path) && Files.isReadable(path) && !Files.isDirectory(path)) { try { return Optional.of(Files.newInputStream(path)); } catch (IOException e) { throw new ConfigException("Failed to read configuration from path: " + path.toAbsolutePath(), e); } } else { return Optional.empty(); } }; } @Override public Optional parser() { return super.parser(); } @Override public Optional mediaType() { return super.mediaType(); } @Override public Optional pollingStrategy() { return super.pollingStrategy(); } @Override public Optional> changeWatcher() { return super.changeWatcher(); } @Override public boolean exists() { return Files.exists(filePath); } /** * File ConfigSource Builder. *

* It allows to configure following properties: *

    *
  • {@code path} - configuration file path;
  • *
  • {@code optional} - is existence of configuration resource optional, or mandatory (by default)?
  • *
  • {@code media-type} - configuration content media type to be used to look for appropriate {@link ConfigParser};
  • *
  • {@code parser} - or directly set {@link ConfigParser} instance to be used to parse the source;
  • *
*

* If {@code media-type} not set it tries to guess it from file extension. */ public static final class Builder extends AbstractConfigSourceBuilder implements PollableSource.Builder, WatchableSource.Builder, ParsableSource.Builder, io.helidon.common.Builder { private Path path; private Builder() { } /** * Configure the path to read configuration from (mandatory). * * @param path path of a file to use * @return updated builder instance */ public Builder path(Path path) { this.path = path; return this; } /** * {@inheritDoc} *

    *
  • {@code path} - path to the file containing the configuration
  • *
* * @param metaConfig configuration properties used to configure a builder instance. * @return modified builder instance */ @Override public Builder config(Config metaConfig) { metaConfig.get(PATH_KEY).as(Path.class).ifPresent(this::path); return super.config(metaConfig); } @Override public Builder parser(ConfigParser parser) { return super.parser(parser); } @Override public Builder mediaType(MediaType mediaType) { return super.mediaType(mediaType); } @Override public Builder changeWatcher(ChangeWatcher changeWatcher) { return super.changeWatcher(changeWatcher); } @Override public Builder pollingStrategy(PollingStrategy pollingStrategy) { return super.pollingStrategy(pollingStrategy); } /** * Builds new instance of File ConfigSource. *

* If {@code media-type} not set it tries to guess it from file extension. * * @return new instance of File ConfigSource. */ @Override public FileConfigSource build() { if (null == path) { throw new IllegalArgumentException("File path cannot be null"); } return new FileConfigSource(this); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy