#1085 Improve of help translation files

- Avoid logging an error if a help_{lang}.yml file does not exist in the JAR
- No longer suggest /authme messages for updating the help translation
- Create consistency test to ensure that all help_{lang}.yml files in the JAR have entries for all help sections / messages / default permissions
This commit is contained in:
ljacqu 2017-02-04 22:00:53 +01:00
parent 9425cc7fc3
commit 20cd9e9588
7 changed files with 124 additions and 10 deletions

View File

@ -1,6 +1,7 @@
package fr.xephi.authme.message; package fr.xephi.authme.message;
import fr.xephi.authme.ConsoleLogger; import fr.xephi.authme.ConsoleLogger;
import fr.xephi.authme.util.FileUtils;
import org.bukkit.configuration.file.FileConfiguration; import org.bukkit.configuration.file.FileConfiguration;
import org.bukkit.configuration.file.YamlConfiguration; import org.bukkit.configuration.file.YamlConfiguration;
@ -16,6 +17,7 @@ public class MessageFileHandler {
// regular file // regular file
private final String filename; private final String filename;
private final FileConfiguration configuration; private final FileConfiguration configuration;
private final String updateAddition;
// default file // default file
private final String defaultFile; private final String defaultFile;
private FileConfiguration defaultConfiguration; private FileConfiguration defaultConfiguration;
@ -25,11 +27,15 @@ public class MessageFileHandler {
* *
* @param file the file to use for messages * @param file the file to use for messages
* @param defaultFile the default file from the JAR to use if no message is found * @param defaultFile the default file from the JAR to use if no message is found
* @param updateCommand command to update the messages file (nullable) to show in error messages
*/ */
public MessageFileHandler(File file, String defaultFile) { public MessageFileHandler(File file, String defaultFile, String updateCommand) {
this.filename = file.getName(); this.filename = file.getName();
this.configuration = YamlConfiguration.loadConfiguration(file); this.configuration = YamlConfiguration.loadConfiguration(file);
this.defaultFile = defaultFile; this.defaultFile = defaultFile;
this.updateAddition = updateCommand == null
? ""
: " (or run " + updateCommand + ")";
} }
/** /**
@ -53,7 +59,7 @@ public class MessageFileHandler {
if (message == null) { if (message == null) {
ConsoleLogger.warning("Error getting message with key '" + key + "'. " ConsoleLogger.warning("Error getting message with key '" + key + "'. "
+ "Please update your config file '" + filename + "' (or run /authme messages)"); + "Please update your config file '" + filename + "'" + updateAddition);
return getDefault(key); return getDefault(key);
} }
return message; return message;
@ -78,7 +84,7 @@ public class MessageFileHandler {
*/ */
private String getDefault(String key) { private String getDefault(String key) {
if (defaultConfiguration == null) { if (defaultConfiguration == null) {
InputStream stream = MessageFileHandler.class.getClassLoader().getResourceAsStream(defaultFile); InputStream stream = FileUtils.getResourceFromJar(defaultFile);
defaultConfiguration = YamlConfiguration.loadConfiguration(new InputStreamReader(stream)); defaultConfiguration = YamlConfiguration.loadConfiguration(new InputStreamReader(stream));
} }
String message = defaultConfiguration.getString(key); String message = defaultConfiguration.getString(key);

View File

@ -36,10 +36,23 @@ public class MessageFileHandlerProvider {
* @return the message file handler * @return the message file handler
*/ */
public MessageFileHandler initializeHandler(Function<String, String> pathBuilder) { public MessageFileHandler initializeHandler(Function<String, String> pathBuilder) {
return initializeHandler(pathBuilder, null);
}
/**
* Initializes a message file handler with the messages file of the configured language.
* Ensures beforehand that the messages file exists or creates it otherwise.
*
* @param pathBuilder function taking the configured language code as argument and returning the messages file
* @param updateCommand command to run to update the languages file (nullable)
* @return the message file handler
*/
public MessageFileHandler initializeHandler(Function<String, String> pathBuilder, String updateCommand) {
String language = settings.getProperty(PluginSettings.MESSAGES_LANGUAGE); String language = settings.getProperty(PluginSettings.MESSAGES_LANGUAGE);
return new MessageFileHandler( return new MessageFileHandler(
initializeFile(language, pathBuilder), initializeFile(language, pathBuilder),
pathBuilder.apply(DEFAULT_LANGUAGE)); pathBuilder.apply(DEFAULT_LANGUAGE),
updateCommand);
} }
/** /**
@ -53,7 +66,8 @@ public class MessageFileHandlerProvider {
File initializeFile(String language, Function<String, String> pathBuilder) { File initializeFile(String language, Function<String, String> pathBuilder) {
String filePath = pathBuilder.apply(language); String filePath = pathBuilder.apply(language);
File file = new File(dataFolder, filePath); File file = new File(dataFolder, filePath);
if (FileUtils.copyFileFromResource(file, filePath)) { // Check that JAR file exists to avoid logging an error
if (FileUtils.getResourceFromJar(filePath) != null && FileUtils.copyFileFromResource(file, filePath)) {
return file; return file;
} }

View File

@ -107,7 +107,7 @@ public class Messages implements Reloadable {
@Override @Override
public void reload() { public void reload() {
this.messageFileHandler = messageFileHandlerProvider this.messageFileHandler = messageFileHandlerProvider
.initializeHandler(lang -> "messages/messages_" + lang + ".yml"); .initializeHandler(lang -> "messages/messages_" + lang + ".yml", "/authme messages");
} }
private static String formatMessage(String message) { private static String formatMessage(String message) {

View File

@ -4,6 +4,7 @@
# ------------------------------------------------------- # -------------------------------------------------------
# Lista de textos usados na seção de ajuda: # Lista de textos usados na seção de ajuda:
common: common:
header: '==========[ Ajuda AuthMeReloaded ]=========='
optional: 'Opcional' optional: 'Opcional'
hasPermission: 'Você tem permissão' hasPermission: 'Você tem permissão'
noPermission: 'Sem Permissão' noPermission: 'Sem Permissão'

View File

@ -39,10 +39,10 @@ public class HelpMessagesServiceTest {
@Mock @Mock
private MessageFileHandlerProvider messageFileHandlerProvider; private MessageFileHandlerProvider messageFileHandlerProvider;
@SuppressWarnings("unchecked")
@BeforeInjecting @BeforeInjecting
@SuppressWarnings("unchecked")
public void initializeHandler() { public void initializeHandler() {
MessageFileHandler handler = new MessageFileHandler(getJarFile(TEST_FILE), "messages/messages_en.yml"); MessageFileHandler handler = new MessageFileHandler(getJarFile(TEST_FILE), "messages/messages_en.yml", null);
given(messageFileHandlerProvider.initializeHandler(any(Function.class))).willReturn(handler); given(messageFileHandlerProvider.initializeHandler(any(Function.class))).willReturn(handler);
} }

View File

@ -0,0 +1,93 @@
package fr.xephi.authme.message;
import fr.xephi.authme.TestHelper;
import fr.xephi.authme.command.help.HelpMessage;
import fr.xephi.authme.command.help.HelpSection;
import fr.xephi.authme.permission.DefaultPermission;
import org.bukkit.configuration.file.FileConfiguration;
import org.bukkit.configuration.file.YamlConfiguration;
import org.hamcrest.Matcher;
import org.junit.Before;
import org.junit.Test;
import java.io.File;
import java.util.Arrays;
import java.util.List;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import static org.hamcrest.Matchers.both;
import static org.hamcrest.Matchers.emptyString;
import static org.hamcrest.Matchers.not;
import static org.hamcrest.Matchers.nullValue;
import static org.junit.Assert.assertThat;
/**
* Tests that all help_xx.yml files contain all entries for
* {@link HelpSection}, {@link HelpMessage} and {@link DefaultPermission}.
*/
public class HelpMessageConsistencyTest {
private static final String MESSAGES_FOLDER = "/messages";
private static final Pattern HELP_MESSAGES_FILE = Pattern.compile("help_[a-z]+\\.yml");
private List<File> helpFiles;
@Before
public void findHelpMessagesFiles() {
File folder = TestHelper.getJarFile(MESSAGES_FOLDER);
File[] files = folder.listFiles();
if (files == null || files.length == 0) {
throw new IllegalStateException("Could not get files from '" + MESSAGES_FOLDER + "'");
}
helpFiles = Arrays.stream(files)
.filter(file -> HELP_MESSAGES_FILE.matcher(file.getName()).matches())
.collect(Collectors.toList());
}
@Test
public void shouldHaveRequiredEntries() {
for (File file : helpFiles) {
// given
FileConfiguration configuration = YamlConfiguration.loadConfiguration(file);
// when / then
assertHasAllHelpSectionEntries(file.getName(), configuration);
}
}
private void assertHasAllHelpSectionEntries(String filename, FileConfiguration configuration) {
for (HelpSection section : HelpSection.values()) {
assertThat(filename + " should have entry for HelpSection '" + section + "'",
configuration.getString(section.getKey()), notEmptyString());
}
for (HelpMessage message : HelpMessage.values()) {
assertThat(filename + " should have entry for HelpMessage '" + message + "'",
configuration.getString(message.getKey()), notEmptyString());
}
for (DefaultPermission defaultPermission : DefaultPermission.values()) {
assertThat(filename + " should have entry for DefaultPermission '" + defaultPermission + "'",
configuration.getString(getPathForDefaultPermission(defaultPermission)), notEmptyString());
}
}
private static String getPathForDefaultPermission(DefaultPermission defaultPermission) {
String path = "common.defaultPermissions.";
switch (defaultPermission) {
case ALLOWED:
return path + "allowed";
case NOT_ALLOWED:
return path + "notAllowed";
case OP_ONLY:
return path + "opOnly";
default:
throw new IllegalStateException("Unknown default permission '" + defaultPermission + "'");
}
}
private static Matcher<String> notEmptyString() {
return both(not(emptyString())).and(not(nullValue()));
}
}

View File

@ -233,8 +233,8 @@ public class MessagesIntegrationTest {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
private static MessageFileHandlerProvider providerReturning(File file, String defaultFile) { private static MessageFileHandlerProvider providerReturning(File file, String defaultFile) {
MessageFileHandlerProvider handler = mock(MessageFileHandlerProvider.class); MessageFileHandlerProvider handler = mock(MessageFileHandlerProvider.class);
given(handler.initializeHandler(any(Function.class))) given(handler.initializeHandler(any(Function.class), anyString()))
.willReturn(new MessageFileHandler(file, defaultFile)); .willReturn(new MessageFileHandler(file, defaultFile, "/authme messages"));
return handler; return handler;
} }
} }