ProvidenceConfigSupplier.java
/*
* Copyright 2016,2017 Providence Authors
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 net.morimekta.providence.config.impl;
import net.morimekta.providence.PMessage;
import net.morimekta.providence.config.ConfigListener;
import net.morimekta.providence.config.ConfigSupplier;
import net.morimekta.providence.config.parser.ConfigException;
import net.morimekta.providence.config.parser.ConfigParser;
import net.morimekta.util.FileWatcher;
import net.morimekta.util.Pair;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.Clock;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
/**
* A supplier to get a config (aka message) from a providence config. This is
* essentially the initiator for the config. It will always have a config
* message instance, and will log (error) if it later fails to load an updated
* config.
*/
public class ProvidenceConfigSupplier<Message extends PMessage<Message>>
extends UpdatingConfigSupplier<Message> {
private static final Logger LOGGER = LoggerFactory.getLogger(ProvidenceConfigSupplier.class);
private final Path configFile;
private final ConfigParser configParser;
private final Set<String> includedFiles;
private final FileWatcher fileWatcher;
private final ConfigListener<Message> configListener;
private final FileWatcher.Listener fileListener;
private final ConfigSupplier<Message> parentSupplier;
public ProvidenceConfigSupplier(@Nonnull Path configFile,
@Nullable ConfigSupplier<Message> parentSupplier,
@Nullable FileWatcher fileWatcher,
@Nonnull ConfigParser configParser,
@Nonnull Clock clock)
throws ConfigException {
super(clock);
this.configFile = configFile;
this.configParser = configParser;
this.parentSupplier = parentSupplier;
this.includedFiles = Collections.synchronizedSet(new HashSet<>());
this.includedFiles.add(configFile.toString());
this.fileWatcher = fileWatcher;
synchronized (this) {
if (fileWatcher != null) {
fileListener = file -> {
if (configFile.equals(file) || includedFiles.contains(file.toString())) {
reload();
}
};
fileWatcher.weakAddWatcher(configFile, fileListener);
} else {
fileListener = null;
}
if (parentSupplier != null) {
this.configListener = config -> this.reload();
this.parentSupplier.addListener(configListener);
set(loadConfig(parentSupplier.get()));
} else {
this.configListener = null;
set(loadConfig(null));
}
}
}
@Override
public String toString() {
StringBuilder builder = new StringBuilder();
builder.append("ProvidenceConfig{")
.append(configFile.getFileName().toString());
if (parentSupplier != null) {
builder.append(", parent=");
builder.append(parentSupplier.getName());
}
builder.append("}");
return builder.toString();
}
@Override
public String getName() {
return "ProvidenceConfig{" + configFile.getFileName() + "}";
}
/**
* Trigger reloading of the config file.
*/
private void reload() {
try {
if (!Files.exists(configFile)) {
LOGGER.warn("Config file deleted " + configFile + ", keeping old config.");
return;
}
LOGGER.trace("Config reload triggered for " + configFile);
if (parentSupplier != null) {
set(loadConfig(parentSupplier.get()));
} else {
set(loadConfig(null));
}
} catch (ConfigException e) {
LOGGER.error("Exception when reloading " + configFile, e);
}
}
@Nonnull
private Message loadConfig(@Nullable Message parent) throws ConfigException {
Pair<Message, Set<String>> tmp = configParser.parseConfig(configFile, parent);
if (fileWatcher != null) {
synchronized (this) {
if (!tmp.second.equals(includedFiles)) {
includedFiles.clear();
includedFiles.addAll(tmp.second);
for (String included : includedFiles) {
fileWatcher.weakAddWatcher(Paths.get(included), fileListener);
}
}
}
}
return tmp.first;
}
}