ResourceBundleMessageResolver.java
package com.gh.mygreen.xlsmapper.localization;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.MissingResourceException;
import java.util.Objects;
import java.util.Optional;
import java.util.ResourceBundle;
import com.gh.mygreen.xlsmapper.util.ArgUtils;
/**
* {@link ResourceBundle}を元にメッセージを解決するクラス。
* <p>クラスパスのルートにリソース名が{@literal SheetValidationMessages}のプロパティファイルを配置していると自動的に読み込みます。</p>
* <p>デフォルトでは、{@link ResourceBundleMessageResolver#DEFAULT_MESSAGE}に配置されているリソースファイルを読み込みます。</p>
*
* @version 2.0
* @author T.TSUCHIE
*
*/
public class ResourceBundleMessageResolver implements MessageResolver {
/**
* デフォルトのメッセージソースのパス
*/
public static final String DEFAULT_MESSAGE = "com.gh.mygreen.xlsmapper.localization.SheetValidationMessages";
private final Map<ResourceBundle, List<String>> messageBundleKeys = new HashMap<ResourceBundle, List<String>>(8);
private final LinkedList<ResourceBundle> messageBundles = new LinkedList<ResourceBundle>();
/**
* メッセージリソースのパスを指定して、インスタンスを作成します。
* @param baseName メッセージリソースのパス。
* @param appendUserResource クラスパスのルートにあるユーザ定義のメッセージソースも読み込むかどうか指定します。
* 引数baseNameの値が {@literal sample.SampleMessages}のとき、クラスパスのルート上にある「SampleMessages」を読み込みます。
* @throws IllegalArgumentException {@literal baseName is null or empty.}
*/
public ResourceBundleMessageResolver(final String baseName, final boolean appendUserResource) {
ArgUtils.notEmpty(baseName, "baseName");
addResourceBundle(ResourceBundle.getBundle(baseName, new EncodingControl("UTF-8")));
// ユーザ定義のリソースを読み込む
if(appendUserResource) {
// リソース名の切り出し
final int index = baseName.lastIndexOf(".");
if(index > 0) {
final String userName = baseName.substring(index+1);
try {
addResourceBundle(ResourceBundle.getBundle(userName, new EncodingControl("UTF-8")));
} catch(Throwable e) { }
}
}
}
/**
* デフォルトのコンストラクタ。
* <p>デフォルトのメッセージソース{@link #DEFAULT_MESSAGE}が自動的に読み込まれます。</p>
*
*/
public ResourceBundleMessageResolver() {
this(DEFAULT_MESSAGE, true);
}
/**
* 独自のメッセージソースを指定してインスタンスを作成する。
* <p>デフォルトのメッセージソース{@link #DEFAULT_MESSAGE}が自動的に読み込まれます。</p>
* @param resourceBundle 独自のメッセージメース
* @throws IllegalArgumentException resourceBundle is null.
*/
public ResourceBundleMessageResolver(final ResourceBundle resourceBundle) {
this();
Objects.requireNonNull(resourceBundle, "resourceBundle should no be null");
addResourceBundle(resourceBundle);
}
/**
* {@inheritDoc}
*/
public Optional<String> getMessage(final String code) {
for(final ResourceBundle bundle : messageBundles) {
final List<String> keys = messageBundleKeys.get(bundle);
if(keys.contains(code)) {
try {
return Optional.of(bundle.getString(code));
} catch(MissingResourceException e) {
return Optional.empty();
}
}
}
return Optional.empty();
}
/**
* メッセージソースを追加します。
* @param resourceBundle 追加するメッセージソース。
* @return 既に追加済みの場合はfalseを返します。
* @throws IllegalArgumentException resourceBundle is null.
*/
public final boolean addResourceBundle(final ResourceBundle resourceBundle) {
Objects.requireNonNull(resourceBundle, "resourceBundle should not be null.");
if(messageBundles.contains(resourceBundle)) {
return false;
}
// 後から追加したリソースを優先するために、先頭に追加する。
messageBundles.addFirst(resourceBundle);
final List<String> keys = new ArrayList<String>();
for(final Enumeration<String> keysEnum = resourceBundle.getKeys(); keysEnum.hasMoreElements();) {
keys.add(keysEnum.nextElement());
}
messageBundleKeys.put(resourceBundle, keys);
return true;
}
/**
* メッセージソースを削除する。
* @param resourceBundle 削除対象のメッセージソース
* @return 登録されているメッセージソースがある場合はtrueを返します。
* @throws IllegalArgumentException resourceBundle is null.
*/
public boolean removeResourceBundle(final ResourceBundle resourceBundle) {
Objects.requireNonNull(resourceBundle, "resourceBundle should not be null.");
if(!messageBundles.contains(resourceBundle)) {
return false;
}
messageBundles.remove(resourceBundle);
return true;
}
}