Adds code for config migration with comment retaining
All checks were successful
KnarCraft/KnarLib/pipeline/head This commit looks good
All checks were successful
KnarCraft/KnarLib/pipeline/head This commit looks good
This commit is contained in:
@@ -0,0 +1,296 @@
|
|||||||
|
package net.knarcraft.knarlib.config;
|
||||||
|
|
||||||
|
import net.knarcraft.knarlib.util.ConfigHelper;
|
||||||
|
import org.bukkit.configuration.ConfigurationSection;
|
||||||
|
import org.bukkit.configuration.InvalidConfigurationException;
|
||||||
|
import org.bukkit.configuration.file.FileConfiguration;
|
||||||
|
import org.bukkit.configuration.file.YamlConfiguration;
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.logging.Level;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A YAML configuration which retains all comments
|
||||||
|
*
|
||||||
|
* <p>This configuration converts all comments to YAML values when loaded, which all start with comment_. When saved,
|
||||||
|
* those YAML values are converted to normal text comments. This ensures that the comments aren't removed by the
|
||||||
|
* YamlConfiguration during its parsing.</p>
|
||||||
|
*
|
||||||
|
* <p>Note: When retrieving a configuration section, that will include comments that start with "comment_". You should
|
||||||
|
* filter those before parsing input, or use "getKeysWithoutComments".</p>
|
||||||
|
*
|
||||||
|
* <p>Also note: This class was originally made for the Stargate rewrite, but can be used for any configuration file.
|
||||||
|
* In order to make sure the comments are retained, always use this class instead of plugin.getConfig() whenever code
|
||||||
|
* might choose to save the configuration.</p>
|
||||||
|
*
|
||||||
|
* @author Kristian Knarvik
|
||||||
|
* @author Thorin
|
||||||
|
*/
|
||||||
|
public class StargateYamlConfiguration extends YamlConfiguration {
|
||||||
|
|
||||||
|
private static final String START_OF_COMMENT_LINE = "[HASHTAG]";
|
||||||
|
private static final String END_OF_COMMENT = "_endOfComment_";
|
||||||
|
private static final String START_OF_COMMENT = "comment_";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@NotNull
|
||||||
|
@Deprecated
|
||||||
|
@SuppressWarnings("deprecation")
|
||||||
|
protected String buildHeader() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@NotNull
|
||||||
|
public String saveToString() {
|
||||||
|
// Convert YAML comments to normal comments
|
||||||
|
return this.convertYAMLMappingsToComments(super.saveToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void loadFromString(@NotNull String contents) throws InvalidConfigurationException {
|
||||||
|
// Convert normal comments to YAML comments to prevent them from disappearing
|
||||||
|
super.loadFromString(this.convertCommentsToYAMLMappings(contents));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads the configuration of the given plugin
|
||||||
|
*
|
||||||
|
* @param plugin <p>The plugin to load the configuration from</p>
|
||||||
|
* @return <p>The loaded configuration, or null if unsuccessful</p>
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@Nullable
|
||||||
|
public static StargateYamlConfiguration loadConfiguration(@NotNull Plugin plugin) {
|
||||||
|
StargateYamlConfiguration configuration = new StargateYamlConfiguration();
|
||||||
|
configuration.options().copyDefaults(true);
|
||||||
|
try {
|
||||||
|
configuration.load(new File(plugin.getDataFolder(), ConfigHelper.getConfigFile()));
|
||||||
|
return configuration;
|
||||||
|
} catch (IOException | InvalidConfigurationException exception) {
|
||||||
|
plugin.getLogger().log(Level.SEVERE, "Unable to load the configuration! Message: " + exception.getMessage());
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves the given configuration
|
||||||
|
*
|
||||||
|
* @param plugin <p>The plugin to save configuration for</p>
|
||||||
|
* @param configuration <p>The configuration to save</p>
|
||||||
|
* @return <p>True if the configuration was successfully saved</p>
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public static boolean saveConfiguration(@NotNull Plugin plugin, @NotNull StargateYamlConfiguration configuration) {
|
||||||
|
try {
|
||||||
|
configuration.save(new File(plugin.getDataFolder(), ConfigHelper.getConfigFile()));
|
||||||
|
return true;
|
||||||
|
} catch (IOException exception) {
|
||||||
|
plugin.getLogger().log(Level.SEVERE, "Unable to save the configuration! Message: " + exception.getMessage());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a configuration section's keys, without any comment entries
|
||||||
|
*
|
||||||
|
* @param configurationSection <p>The configuration section to get keys for</p>
|
||||||
|
* @param deep <p>Whether to get keys for child elements as well</p>
|
||||||
|
* @return <p>The configuration section's keys, with comment entries removed</p>
|
||||||
|
*/
|
||||||
|
public static Set<String> getKeysWithoutComments(@NotNull ConfigurationSection configurationSection, boolean deep) {
|
||||||
|
Set<String> keys = new HashSet<>(configurationSection.getKeys(deep));
|
||||||
|
keys.removeIf(key -> key.matches(START_OF_COMMENT + "[0-9]+"));
|
||||||
|
return keys;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads a file with comments, and recreates them into yaml mappings
|
||||||
|
*
|
||||||
|
* <p>A mapping follows this format: comment_{CommentNumber}: "The comment"
|
||||||
|
* This needs to be done as comments otherwise get removed using
|
||||||
|
* the {@link FileConfiguration#save(File)} method. The config
|
||||||
|
* needs to be saved if a config value has changed.</p>
|
||||||
|
*
|
||||||
|
* @param configString <p>The config string to convert</p>
|
||||||
|
*/
|
||||||
|
@NotNull
|
||||||
|
private String convertCommentsToYAMLMappings(@NotNull String configString) {
|
||||||
|
StringBuilder yamlBuilder = new StringBuilder();
|
||||||
|
List<String> currentComment = new ArrayList<>();
|
||||||
|
int commentId = 0;
|
||||||
|
int previousIndentation = 0;
|
||||||
|
|
||||||
|
for (String line : configString.split("\n")) {
|
||||||
|
String trimmed = line.trim();
|
||||||
|
if (trimmed.startsWith("#")) {
|
||||||
|
// Store the indentation of the block
|
||||||
|
if (currentComment.isEmpty()) {
|
||||||
|
previousIndentation = getIndentation(line);
|
||||||
|
}
|
||||||
|
//Temporarily store the comment line
|
||||||
|
addComment(currentComment, trimmed);
|
||||||
|
} else {
|
||||||
|
addYamlString(yamlBuilder, currentComment, line, previousIndentation, commentId);
|
||||||
|
commentId++;
|
||||||
|
previousIndentation = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return yamlBuilder.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds a YAML string to the given string builder
|
||||||
|
*
|
||||||
|
* @param yamlBuilder <p>The string builder used for building YAML</p>
|
||||||
|
* @param currentComment <p>The comment to add as a YAML string</p>
|
||||||
|
* @param line <p>The current line</p>
|
||||||
|
* @param previousIndentation <p>The indentation of the current block comment</p>
|
||||||
|
* @param commentId <p>The id of the comment</p>
|
||||||
|
*/
|
||||||
|
private void addYamlString(@NotNull StringBuilder yamlBuilder, @NotNull List<String> currentComment,
|
||||||
|
@NotNull String line, int previousIndentation, int commentId) {
|
||||||
|
String trimmed = line.trim();
|
||||||
|
//Write the full formatted comment to the StringBuilder
|
||||||
|
if (!currentComment.isEmpty()) {
|
||||||
|
int indentation = trimmed.isEmpty() ? previousIndentation : getIndentation(line);
|
||||||
|
generateCommentYAML(yamlBuilder, currentComment, commentId, indentation);
|
||||||
|
currentComment.clear();
|
||||||
|
}
|
||||||
|
//Add the non-comment line assuming it isn't empty
|
||||||
|
if (!trimmed.isEmpty()) {
|
||||||
|
yamlBuilder.append(line).append("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds the given comment to the given list
|
||||||
|
*
|
||||||
|
* @param commentParts <p>The list to add to</p>
|
||||||
|
* @param comment <p>The comment to add</p>
|
||||||
|
*/
|
||||||
|
private void addComment(@NotNull List<String> commentParts, @NotNull String comment) {
|
||||||
|
if (comment.startsWith("# ")) {
|
||||||
|
commentParts.add(comment.replaceFirst("# ", START_OF_COMMENT_LINE));
|
||||||
|
} else {
|
||||||
|
commentParts.add(comment.replaceFirst("#", START_OF_COMMENT_LINE));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates a YAML-compatible string for one comment block
|
||||||
|
*
|
||||||
|
* @param yamlBuilder <p>The string builder to add the generated YAML to</p>
|
||||||
|
* @param commentLines <p>The lines of the comment to convert into YAML</p>
|
||||||
|
* @param commentId <p>The unique id of the comment</p>
|
||||||
|
* @param indentation <p>The indentation to add to every line</p>
|
||||||
|
*/
|
||||||
|
private void generateCommentYAML(@NotNull StringBuilder yamlBuilder, @NotNull List<String> commentLines,
|
||||||
|
int commentId, int indentation) {
|
||||||
|
String subIndentation = this.addIndentation(indentation + 2);
|
||||||
|
//Add the comment start marker
|
||||||
|
yamlBuilder.append(this.addIndentation(indentation)).append(START_OF_COMMENT).append(commentId).append(": |\n");
|
||||||
|
for (String commentLine : commentLines) {
|
||||||
|
//Add each comment line with the proper indentation
|
||||||
|
yamlBuilder.append(subIndentation).append(commentLine).append("\n");
|
||||||
|
}
|
||||||
|
//Add the comment end marker
|
||||||
|
yamlBuilder.append(subIndentation).append(subIndentation).append(END_OF_COMMENT).append("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts the internal YAML mapping format to a readable config file
|
||||||
|
*
|
||||||
|
* <p>The internal YAML structure is converted to a string with the same format as a standard configuration file.
|
||||||
|
* The internal structure has comments in the format: START_OF_COMMENT + id + multi-line YAML string +
|
||||||
|
* END_OF_COMMENT.</p>
|
||||||
|
*
|
||||||
|
* @param yamlString <p>A string using the YAML format</p>
|
||||||
|
* @return <p>The corresponding comment string</p>
|
||||||
|
*/
|
||||||
|
@NotNull
|
||||||
|
private String convertYAMLMappingsToComments(@NotNull String yamlString) {
|
||||||
|
StringBuilder finalText = new StringBuilder();
|
||||||
|
|
||||||
|
String[] lines = yamlString.split("\n");
|
||||||
|
for (int currentIndex = 0; currentIndex < lines.length; currentIndex++) {
|
||||||
|
String line = lines[currentIndex];
|
||||||
|
String possibleComment = line.trim();
|
||||||
|
|
||||||
|
if (possibleComment.startsWith(START_OF_COMMENT)) {
|
||||||
|
//Add an empty line before every comment block
|
||||||
|
finalText.append("\n");
|
||||||
|
currentIndex = readComment(finalText, lines, currentIndex + 1, getIndentation(line));
|
||||||
|
} else {
|
||||||
|
//Output the configuration key
|
||||||
|
finalText.append(line).append("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return finalText.toString().trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fully reads a comment
|
||||||
|
*
|
||||||
|
* @param builder <p>The string builder to write to</p>
|
||||||
|
* @param lines <p>The lines to read from</p>
|
||||||
|
* @param startIndex <p>The index to start reading from</p>
|
||||||
|
* @param commentIndentation <p>The indentation of the read comment</p>
|
||||||
|
* @return <p>The index containing the next non-comment line</p>
|
||||||
|
*/
|
||||||
|
private int readComment(@NotNull StringBuilder builder, @NotNull String[] lines, int startIndex,
|
||||||
|
int commentIndentation) {
|
||||||
|
for (int currentIndex = startIndex; currentIndex < lines.length; currentIndex++) {
|
||||||
|
String line = lines[currentIndex];
|
||||||
|
String possibleComment = line.trim();
|
||||||
|
if (!line.contains(END_OF_COMMENT)) {
|
||||||
|
possibleComment = possibleComment.replace(START_OF_COMMENT_LINE, "");
|
||||||
|
builder.append(addIndentation(commentIndentation)).append("# ").append(possibleComment).append("\n");
|
||||||
|
} else {
|
||||||
|
return currentIndex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return startIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a string containing the given indentation
|
||||||
|
*
|
||||||
|
* @param indentationSpaces <p>The number spaces to use for indentation</p>
|
||||||
|
* @return <p>A string containing the number of spaces specified</p>
|
||||||
|
*/
|
||||||
|
@NotNull
|
||||||
|
private String addIndentation(int indentationSpaces) {
|
||||||
|
return " ".repeat(Math.max(0, indentationSpaces));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the indentation (number of spaces) of the given line
|
||||||
|
*
|
||||||
|
* @param line <p>The line to get indentation of</p>
|
||||||
|
* @return <p>The number of spaces in the line's indentation</p>
|
||||||
|
*/
|
||||||
|
private int getIndentation(@NotNull String line) {
|
||||||
|
int spacesFound = 0;
|
||||||
|
for (char aCharacter : line.toCharArray()) {
|
||||||
|
if (aCharacter == ' ') {
|
||||||
|
spacesFound++;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return spacesFound;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
162
src/main/java/net/knarcraft/knarlib/util/ConfigHelper.java
Normal file
162
src/main/java/net/knarcraft/knarlib/util/ConfigHelper.java
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
package net.knarcraft.knarlib.util;
|
||||||
|
|
||||||
|
import net.knarcraft.knarlib.config.StargateYamlConfiguration;
|
||||||
|
import net.knarcraft.knarlib.property.ColorConversion;
|
||||||
|
import org.bukkit.configuration.ConfigurationSection;
|
||||||
|
import org.bukkit.configuration.InvalidConfigurationException;
|
||||||
|
import org.bukkit.configuration.MemorySection;
|
||||||
|
import org.bukkit.configuration.file.FileConfiguration;
|
||||||
|
import org.bukkit.configuration.file.YamlConfiguration;
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.logging.Level;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A helper class for dealing with a plugin's configuration file
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public final class ConfigHelper {
|
||||||
|
|
||||||
|
private static final String CONFIG_FILE = "config.yml";
|
||||||
|
private static final String BACKUP_CONFIG_FILE = "config.yml.old";
|
||||||
|
private static final String MIGRATION_FILE = "config-migrations.txt";
|
||||||
|
|
||||||
|
private ConfigHelper() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the relative string path of the configuration file
|
||||||
|
*
|
||||||
|
* @return <p>The configuration file path</p>
|
||||||
|
*/
|
||||||
|
public static String getConfigFile() {
|
||||||
|
return CONFIG_FILE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves any missing configuration values to the given plugin's configuration
|
||||||
|
*
|
||||||
|
* @param plugin <p>The plugin to add missing default values to</p>
|
||||||
|
*/
|
||||||
|
public static void saveDefaults(@NotNull Plugin plugin) {
|
||||||
|
plugin.saveDefaultConfig();
|
||||||
|
plugin.getConfig().options().copyDefaults(true);
|
||||||
|
plugin.reloadConfig();
|
||||||
|
plugin.saveConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes all configuration values from the old name to the new name
|
||||||
|
*
|
||||||
|
* <p>Note: This method expects a file "config-migrations.txt" in the resources directory that contains mappings:
|
||||||
|
* oldKey=replacementKey in order to migrate from old to new configuration values.
|
||||||
|
* The old configuration file will be saved to config.yml.old</p>
|
||||||
|
*
|
||||||
|
* @param plugin <p>The plugin to migrate the configuration for</p>
|
||||||
|
* @return <p>True if the migration succeeded without any issues</p>
|
||||||
|
*/
|
||||||
|
public static boolean migrateConfig(@NotNull Plugin plugin) {
|
||||||
|
File dataFolder = plugin.getDataFolder();
|
||||||
|
//Save the old config just in case something goes wrong
|
||||||
|
try {
|
||||||
|
StargateYamlConfiguration currentConfiguration = new StargateYamlConfiguration();
|
||||||
|
currentConfiguration.load(new File(dataFolder, CONFIG_FILE));
|
||||||
|
currentConfiguration.save(new File(dataFolder, BACKUP_CONFIG_FILE));
|
||||||
|
} catch (IOException | InvalidConfigurationException exception) {
|
||||||
|
plugin.getLogger().log(Level.WARNING, "Unable to save old backup and do migration");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Load old and new configuration
|
||||||
|
plugin.reloadConfig();
|
||||||
|
FileConfiguration oldConfiguration = plugin.getConfig();
|
||||||
|
InputStream configStream = FileHelper.getInputStreamForInternalFile("/" + CONFIG_FILE);
|
||||||
|
if (configStream == null) {
|
||||||
|
plugin.getLogger().log(Level.SEVERE, "Could not migrate the configuration, as the internal " +
|
||||||
|
"configuration could not be read!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
YamlConfiguration newConfiguration = StargateYamlConfiguration.loadConfiguration(
|
||||||
|
FileHelper.getBufferedReaderFromInputStream(configStream));
|
||||||
|
|
||||||
|
//Read all available config migrations
|
||||||
|
Map<String, String> migrationFields;
|
||||||
|
try {
|
||||||
|
InputStream migrationStream = FileHelper.getInputStreamForInternalFile("/" + MIGRATION_FILE);
|
||||||
|
if (migrationStream == null) {
|
||||||
|
plugin.getLogger().log(Level.SEVERE, "Could not migrate the configuration, as the internal " +
|
||||||
|
"migration paths could not be read!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
migrationFields = FileHelper.readKeyValuePairs(FileHelper.getBufferedReaderFromInputStream(migrationStream),
|
||||||
|
"=", ColorConversion.NONE);
|
||||||
|
} catch (IOException exception) {
|
||||||
|
plugin.getLogger().log(Level.WARNING, "Unable to load config migration file");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
//Replace old config names with the new ones
|
||||||
|
for (String key : migrationFields.keySet()) {
|
||||||
|
if (oldConfiguration.contains(key)) {
|
||||||
|
migrateProperty(migrationFields, key, oldConfiguration);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy all keys to the new config
|
||||||
|
for (String key : StargateYamlConfiguration.getKeysWithoutComments(oldConfiguration, true)) {
|
||||||
|
if (oldConfiguration.get(key) instanceof MemorySection) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
newConfiguration.set(key, oldConfiguration.get(key));
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
newConfiguration.save(new File(dataFolder, CONFIG_FILE));
|
||||||
|
} catch (IOException exception) {
|
||||||
|
plugin.getLogger().log(Level.WARNING, "Unable to save migrated config");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
plugin.reloadConfig();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Migrates one configuration property
|
||||||
|
*
|
||||||
|
* @param migrationFields <p>The configuration fields to be migrated</p>
|
||||||
|
* @param key <p>The key/path of the property to migrate</p>
|
||||||
|
* @param oldConfiguration <p>The original pre-migration configuration</p>
|
||||||
|
*/
|
||||||
|
private static void migrateProperty(@NotNull Map<String, String> migrationFields, @NotNull String key,
|
||||||
|
@NotNull FileConfiguration oldConfiguration) {
|
||||||
|
String newPath = migrationFields.get(key);
|
||||||
|
Object oldValue = oldConfiguration.get(key);
|
||||||
|
if (!newPath.trim().isEmpty()) {
|
||||||
|
if (oldConfiguration.isConfigurationSection(key)) {
|
||||||
|
// Copy each value of a configuration section
|
||||||
|
ConfigurationSection sourceSection = oldConfiguration.getConfigurationSection(key);
|
||||||
|
ConfigurationSection destinationSection = oldConfiguration.createSection(newPath);
|
||||||
|
if (sourceSection == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (String path : StargateYamlConfiguration.getKeysWithoutComments(sourceSection, true)) {
|
||||||
|
destinationSection.set(path, sourceSection.get(path));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Copy the value to the new path
|
||||||
|
oldConfiguration.set(newPath, oldValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the old path's value
|
||||||
|
oldConfiguration.set(key, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Reference in New Issue
Block a user