nablarch.test.core.reader.TestDataParsingTemplate Maven / Gradle / Ivy
package nablarch.test.core.reader;
import nablarch.core.util.StringUtil;
import nablarch.test.NablarchTestUtils;
import nablarch.test.core.util.interpreter.InterpretationContext;
import nablarch.test.core.util.interpreter.TestDataInterpreter;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import static nablarch.core.util.Builder.concat;
/**
* テストデータ解析用のテンプレートクラス。
*
* @param 解析後の型
* @author T.Kawasaki
*/
abstract class TestDataParsingTemplate {
/** 解析に使用するリーダ */
private final TestDataReader reader;
/** 委譲先の{@link nablarch.test.core.util.interpreter.TestDataInterpreter} */
private final List interpreters;
/** 解析対象のデータ型 */
private final DataType targetType;
/** テストデータのキャッシュ */
private static final Map>> TEST_DATA_CACHE = NablarchTestUtils.createLRUMap(8);
/** テストデータ */
private List> testData;
/** テストデータを読み込む際のインデックス */
private int index;
/** ディレクトリ */
protected String directory; // SUPPRESS CHECKSTYLE サブクラスに対してカプセル化が不要なため
/** リソース名 */
protected String resource; // SUPPRESS CHECKSTYLE サブクラスに対してカプセル化が不要なため
/**
* 行読み込み時に起動されるコールバックメソッド。
*
* @param line 読み込まれた行データ
*/
abstract void onReadLine(List line);
/**
* 処理対象のデータ型を発見した場合に起動されるコールバックメソッド。
*
* @param line データ型の定義を含む行データ
*/
abstract void onTargetTypeFound(List line);
/**
* 処理対象のデータ型であるかどうか判定する。
*
* @param line 読み込み中の行データ
* @param id ID
* @return 処理対象のデータ型である場合、真
*/
abstract boolean isTargetType(List line, String id);
/**
* 次の処理対象のデータを発見した場合に処理を停止するか否かを判定する。
*
* @return 次の処理対象のデータを発見した場合に処理停止する場合は、真
*/
abstract boolean shouldStopOnNextOne();
/**
* 解析結果を返却する。
*
* @return 解析結果
*/
abstract RET getResult();
/**
* コンストラクタ。
*
* @param reader 解析に使用するリーダ
* @param interpreters 解釈クラス
* @param targetType 解析対象のデータ型
*/
TestDataParsingTemplate(TestDataReader reader, List interpreters, DataType targetType) {
this.reader = reader;
this.interpreters = interpreters;
this.targetType = targetType;
}
/**
* 解析対象のデータ型を返却する。
*
* @return 解析対象のデータ型
*/
protected final DataType getTargetType() {
return targetType;
}
/**
* 解析を実行する。
*
* 読み込んだシートはキャッシュする。
*
* @param directory 読み込み元のディレクトリ
* @param resource リソース名称
* @param id ID
*/
public final void parse(String directory, String resource, String id) {
parse(directory, resource, id, true);
}
/**
* 解析を実行する。
* @param directory 読み込み元のディレクトリ
* @param resource リソース名称
* @param id ID
* @param saveCache 読み込んだファイルのデータをキャッシュに保存するか
*/
public final void parse(String directory, String resource, String id, boolean saveCache) {
this.directory = directory;
this.resource = resource;
String dataCacheKey = directory + '/' + resource;
if (!TEST_DATA_CACHE.containsKey(dataCacheKey)) {
// テストデータがキャッシュ上に存在しない場合は、テストデータをロードしキャッシュする。
if (reader instanceof PoiXlsReader) {
((PoiXlsReader) reader).setUseCache(saveCache);
}
reader.open(directory, resource);
try {
testData = readTestData();
if (saveCache) {
TEST_DATA_CACHE.put(dataCacheKey, testData);
}
} finally {
reader.close();
}
} else {
testData = TEST_DATA_CACHE.get(dataCacheKey);
}
try {
index = 0;
parse(id);
} catch (RuntimeException e) {
String msg = concat("can't get data. ",
"directory=[", directory, "] resource=[", resource, "] id=[", id, "]");
throw new IllegalStateException(msg, e);
}
}
/**
* テストデータを読み込む。
*
* @return 読み込んだテストデータ
*/
private List> readTestData() {
List> lines = new ArrayList>(1024);
while (true) {
List line = reader.readLine();
if (line == null) {
break;
}
if (isCommentRow(line)) {
continue;
}
line = cutComment(line);
if (isBlankLine(line)) {
continue;
}
List interpret = interpret(line);
// キャッシュするので、他の箇所で書き換えられないようにする。
List unmodifiable = Collections.unmodifiableList(interpret);
lines.add(unmodifiable);
}
// キャッシュするので、他の箇所で書き換えられないようにする。
return Collections.unmodifiableList(lines);
}
/**
* 解析を実行する。
*
* @param id ID
*/
void parse(String id) {
boolean nowReading = false; // データ行読み込み中か
List line;
while ((line = readLine()) != null) {
// テーブル名の行
String firstCol = line.get(0);
DataType dataType = getDataType(firstCol);
// データタイプとグループIDが一致するか
if (isTargetType(line, id)) {
if (nowReading && shouldStopOnNextOne()) {
break;
}
nowReading = true;
onTargetTypeFound(line);
} else if (dataType.equals(DataType.DEFAULT)) {
// データ行読み込み中か?
if (nowReading) {
onReadLine(line);
}
} else {
// データ行読み込み中なら処理終了
if (nowReading) {
break;
}
}
}
}
/** データタイプのリスト */
private static final DataType[] DATA_TYPES = DataType.values();
/**
* データタイプを返却する。
*
* @param dataTypeCell データタイプが記載されたセル
* @return データタイプ
*/
protected final DataType getDataType(String dataTypeCell) {
if (dataTypeCell == null) {
return DataType.DEFAULT;
}
for (DataType type : DATA_TYPES) {
if (dataTypeCell.startsWith(type.getName())) {
return type;
}
}
return DataType.DEFAULT;
}
/**
* テストデータタイプのバリュー値を取得する。
*
* @param dataTypeRow 1行分のデータ
* @return バリュー値
*/
protected final String getTypeValue(List dataTypeRow) {
String str = dataTypeRow.get(0);
return str.substring(str.indexOf('=') + 1);
}
/**
* テストデータから1行文のデータを読み込む。
* テストデータをすべて読み込んだ場合は、nullを返却する。
*
* @return 1行データ
*/
protected final List readLine() {
if (index < testData.size()) {
return testData.get(index++);
}
return null;
}
/** コメントを表す文字列 */
private static final String COMMENT_EXPRESSION = "//";
/**
* コメント行であるか判定する。
* 先頭が//始まりの場合はコメント行とみなす。
*
* @param line 行データ
* @return 判定結果
*/
boolean isCommentRow(List line) {
String first = line.get(0);
return isComment(first);
}
/**
* コメントセルであるかどうか判定する。
*
* @param cell セルの値
* @return そのセルの値が//からスタートしている場合、真
*/
private boolean isComment(String cell) {
return cell != null && cell.startsWith(COMMENT_EXPRESSION);
}
/**
* 1行を表すデータからコメントを削除する。
* コメント部は、「//」以降の要素
*
* @param src 1行を表すテストデータ
* @return コメント部を取り除いた1行データ
*/
private List cutComment(List src) {
List result = new ArrayList(src.size());
for (String s : src) {
if (isComment(s)) {
break;
}
result.add(s);
}
return result;
}
/**
* 全要素が空かどうか判定
*
* @param line 1行分のデータ
* @return 配列の全要素が空であればtrue、空でなければfalse
*/
private boolean isBlankLine(List line) {
return StringUtil.isNullOrEmpty(line);
}
/**
* テストデータ記法を解釈する。
*
* @param originalLine 元のデータ行
* @return 解釈後のデータ行
*/
private List interpret(List originalLine) {
List result = new ArrayList(originalLine.size());
for (String e : originalLine) {
InterpretationContext context
= new InterpretationContext(e, interpreters);
String interpreted = context.invokeNext();
result.add(interpreted);
}
return result;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy