Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.dbflute.helper.dfmap.DfMapStyle Maven / Gradle / Ivy
/*
* Copyright 2014-2019 the original author or authors.
*
* Licensed 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 org.dbflute.helper.dfmap;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import org.dbflute.helper.dfmap.exception.DfMapDuplicateEntryException;
import org.dbflute.helper.dfmap.exception.DfMapParseFailureException;
import org.dbflute.helper.message.ExceptionMessageBuilder;
/**
* The parser for DBFlute map style.
*
* e.g. map-string as one liner
* map:{key1=value1;key2=list:{value21;value22;value23},key3=map:{key31=value31}}
*
* e.g. map-string as formatted
* map:{
* ; key1 = value1
* ; key2 = list:{ value21 ; value22 ; value23 }
* ; key3 = map:{ key31 ; value31 }
* }
*
* e.g. list-string
* list:{key1=value1,key2=list:{value21,value22,value23},key3=map:{key31=value31}}
*
*
* Detail Specification:
*
* o all elements are trimmed e.g. map:{ sea = land } to {sea=land}
* o empty or null value is treated as null e.g. map:{ sea = ; land = null } to both null
* o empty elements are ignored e.g. list:{ ; sea ; ; land } to [sea, land]
* o can escape control marks e.g. '{', ';' by escape character '\'
* o but cannot escape escape character itself
* o basically cannot change control mark, but you can override...
* o can check duplicate entry of map key by option checkDuplicateEntry()
* o can print as one liner by option printOneLiner()
* o non-thread-safe, uses temporary variables
* o but can recycle in same thread
*
* @author jflute
* @since 1.1.8 (2018/05/05 Saturday)
*/
public class DfMapStyle { // migrated MapListString, basically keeping compatible
// this original code (MapListString) was written when jflute was very young
// and migrated to this class with small refactoring for performance
// all lines are very memorable...
// ===================================================================================
// Definition
// ==========
/** The object prefix of map. */
public static final String MAP_PREFIX = "map:";
/** The object prefix of list. */
public static final String LIST_PREFIX = "list:";
/** The control mark of begin-brace. */
public static final String BEGIN_BRACE = "{";
/** The control mark of end-brace. */
public static final String END_BRACE = "}";
/** The control mark of element-delimiter. */
public static final String ELEMENT_DELIMITER = ";";
/** The control mark of value-equal. */
public static final String VALUE_EQUAL = "=";
/** The escape character for control marks. */
protected static final String ESCAPE_CHAR = "\\";
/** The temporary mark of escaped escape character. */
protected static final String ESCAPED_ESCAPE_MARK = "$$df:escapedEscape$$";
// ===================================================================================
// Attribute
// =========
// -----------------------------------------------------
// Object Prefix
// -------------
/** The object prefix of map. e.g. "map:" (NotNull) */
protected final String _mapPrefix;
/** The object prefix of list. e.g. "list:" (NotNull) */
protected final String _listPrefix;
// -----------------------------------------------------
// Control Mark
// ------------
/** The control mark of begin-brace. e.g. "{" (NotNull) */
protected final String _beginBrace;
/** The control mark of end-brace. e.g. "}" (NotNull) */
protected final String _endBrace;
/** The control mark of element-delimiter. e.g. ";" (NotNull) */
protected final String _elementDelimiter;
/** The control mark of value-equal. e.g. "=" (NotNull) */
protected final String _valueEqual;
/** The escape character for control marks. "\\" (NotNull) */
protected final String _escapeChar;
// -----------------------------------------------------
// Cached Expression
// -----------------
/** The cached expression of begin-brace for map. e.g. "map:{" */
protected final String _mappingBeginBrace;
/** The cached expression of begin-brace for list. e.g. "list:{" */
protected final String _listingBeginBrace;
// -----------------------------------------------------
// Temporary
// ---------
/** The whole map-string of logical remainder as temporary variable for parsing process. (NullAllowed: when no process) */
protected String _remainderString;
/** The current index of remainder string for parsing process. */
protected int _currentRemainderIndex;
// -----------------------------------------------------
// Option
// ------
/** Does it check duplicate entry of map for generation? */
protected boolean _checkDuplicateEntry;
/** Does it print as one-liner for building? */
protected boolean _printOneLiner;
// ===================================================================================
// Constructor
// ===========
/**
* Constructor for settings as default.
*/
public DfMapStyle() {
this(MAP_PREFIX, LIST_PREFIX, BEGIN_BRACE, END_BRACE, ELEMENT_DELIMITER, VALUE_EQUAL, ESCAPE_CHAR);
}
// you can use as public in your constructor of sub-class
protected DfMapStyle(String mapPrefix, String listPrefix, String beginBrace, String endBrace, String elementDelimiter,
String valueEqual, String escapeChar) {
assertStringNotNullAndNotTrimmedEmpty("mapPrefix", mapPrefix);
assertStringNotNullAndNotTrimmedEmpty("listPrefix", listPrefix);
assertStringNotNullAndNotTrimmedEmpty("beginBrace", beginBrace);
assertStringNotNullAndNotTrimmedEmpty("endBrace", endBrace);
assertStringNotNullAndNotTrimmedEmpty("delimiter", elementDelimiter);
assertStringNotNullAndNotTrimmedEmpty("valueEqual", valueEqual);
assertStringNotNullAndNotTrimmedEmpty("escapeChar", escapeChar);
_mapPrefix = mapPrefix;
_listPrefix = listPrefix;
_beginBrace = beginBrace;
_endBrace = endBrace;
_elementDelimiter = elementDelimiter;
_valueEqual = valueEqual;
_escapeChar = escapeChar;
_mappingBeginBrace = _mapPrefix + _beginBrace;
_listingBeginBrace = _listPrefix + _beginBrace;
}
// -----------------------------------------------------
// Option
// ------
/**
* Check duplicate entry of map for generation. (throws exception if found)
* @return this. (NotNull)
*/
public DfMapStyle checkDuplicateEntry() {
_checkDuplicateEntry = true;
return this;
}
/**
* Print as one liner for building.
* @return this. (NotNull)
*/
public DfMapStyle printOneLiner() {
_printOneLiner = true;
return this;
}
// ===================================================================================
// Object to String
// ================
// -----------------------------------------------------
// to MapString
// ------------
/**
* Convert to map-string from the map object.
* @param map The map object that has string keys. (NotNull, EmptyAllowed)
* @return The string as map expression. (NotNull)
*/
public String toMapString(Map map) {
assertObjectNotNull("map", map);
final StringBuilder sb = new StringBuilder();
@SuppressWarnings("unchecked")
final Map casted = (Map) map;
final boolean printOneLiner = isPrintOneLiner();
buildMapString(sb, casted, printOneLiner, "", printOneLiner ? "" : " ");
return sb.toString();
}
protected void buildMapString(StringBuilder sb, Map map, boolean printOneLiner, String previousIndent,
String currentIndent) {
doBuildMapStringBegin(sb, map, printOneLiner, previousIndent, currentIndent);
if (!map.isEmpty()) {
int index = 0;
for (Entry entry : map.entrySet()) {
doBuildMapStringCurrentEntry(sb, printOneLiner, previousIndent, currentIndent, index, entry.getKey(), entry.getValue());
++index;
}
if (printOneLiner) {
sb.append(" ");
} else {
sb.append(ln()).append(previousIndent);
}
}
doBuildMapStringEnd(sb, map, printOneLiner, previousIndent, currentIndent);
}
protected void doBuildMapStringBegin(StringBuilder sb, Map map, boolean printOneLiner, String previousIndent,
String currentIndent) {
sb.append(_mapPrefix).append(_beginBrace);
}
protected void doBuildMapStringCurrentEntry(StringBuilder sb, boolean printOneLiner, String previousIndent, String currentIndent,
int index, String key, Object value) {
if (printOneLiner) {
if (index > 0) {
sb.append(" ").append(_elementDelimiter);
}
} else {
sb.append(ln()).append(currentIndent).append(_elementDelimiter);
}
sb.append(" ").append(escapeControlMark(key)).append(" ").append(_valueEqual).append(" ");
if (value instanceof Map, ?>) {
@SuppressWarnings("unchecked")
final Map valueMap = (Map) value;
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildMapString(sb, valueMap, printOneLiner, currentIndent, nextIndent);
} else if (value instanceof List>) {
@SuppressWarnings("unchecked")
final List valueList = (List) value;
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildListString(sb, valueList, printOneLiner, currentIndent, nextIndent);
} else {
final Map resolvedMap = resolvePotentialMapOfBuildingMapString(value);
if (resolvedMap != null) {
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildMapString(sb, resolvedMap, printOneLiner, currentIndent, nextIndent);
} else {
sb.append(escapeControlMark(value));
}
}
}
protected void doBuildMapStringEnd(StringBuilder sb, Map map, boolean printOneLiner, String previousIndent,
String currentIndent) {
sb.append(_endBrace);
}
// -----------------------------------------------------
// to ListString
// -------------
/**
* Convert to list-string from the list object.
* @param list The list object that has object elements. (NotNull, EmptyAllowed)
* @return The string as list expression. (NotNull)
*/
public String toListString(List extends Object> list) {
assertObjectNotNull("list", list);
final StringBuilder sb = new StringBuilder();
@SuppressWarnings("unchecked")
final List casted = (List) list;
final boolean printOneLiner = isPrintOneLiner();
buildListString(sb, casted, printOneLiner, "", printOneLiner ? "" : " ");
return sb.toString();
}
protected void buildListString(StringBuilder sb, List extends Object> list, boolean printOneLiner, String previousIndent,
String currentIndent) {
doBuildListStringBegin(sb, list, printOneLiner, previousIndent, currentIndent);
if (!list.isEmpty()) {
int index = 0;
for (Object value : list) {
doBuildListStringCurrentElement(sb, printOneLiner, previousIndent, currentIndent, index, value);
++index;
}
if (printOneLiner) {
sb.append(" ");
} else {
sb.append(ln()).append(previousIndent);
}
}
doBuildListStringEnd(sb, list, printOneLiner, previousIndent, currentIndent);
}
protected void doBuildListStringBegin(StringBuilder sb, List extends Object> list, boolean printOneLiner, String previousIndent,
String currentIndent) {
sb.append(_listPrefix).append(_beginBrace);
}
protected void doBuildListStringCurrentElement(StringBuilder sb, boolean printOneLiner, String previousIndent, String currentIndent,
int index, Object value) {
if (printOneLiner) {
if (index > 0) {
sb.append(" ").append(_elementDelimiter);
}
} else {
sb.append(ln()).append(currentIndent).append(_elementDelimiter);
}
sb.append(" ");
if (value instanceof Map, ?>) {
@SuppressWarnings("unchecked")
final Map valueMap = (Map) value;
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildMapString(sb, valueMap, printOneLiner, currentIndent, nextIndent);
} else if (value instanceof List>) {
@SuppressWarnings("unchecked")
final List valueList = (List) value;
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildListString(sb, valueList, printOneLiner, currentIndent, nextIndent);
} else {
final Map resolvedMap = resolvePotentialMapOfBuildingMapString(value);
if (resolvedMap != null) {
final String nextIndent = deriveNextIndentOfBuildingMapString(printOneLiner, previousIndent, currentIndent);
buildMapString(sb, resolvedMap, printOneLiner, currentIndent, nextIndent);
} else {
sb.append(escapeControlMark(value));
}
}
}
protected void doBuildListStringEnd(StringBuilder sb, List extends Object> list, boolean printOneLiner, String previousIndent,
String currentIndent) {
sb.append(_endBrace);
}
// -----------------------------------------------------
// Assist Logic
// ------------
protected String deriveNextIndentOfBuildingMapString(boolean printOneLiner, String previousIndent, String currentIndent) {
return printOneLiner ? "" : generally_calculateNextIndent(previousIndent, currentIndent);
}
protected Map resolvePotentialMapOfBuildingMapString(Object value) { // value may be null
// you can override for your bean that can be map like this:
// e.g.
// if (value instanceof Something) {
// return ((Something) value).toMap();
// } else {
// return super.resolvePotentialMapOfBuildingMapString(value);
// }
return null; // returning null means non-map value
}
// ===================================================================================
// Object from String
// ==================
// -----------------------------------------------------
// from MapString
// --------------
/**
* Convert to map object from the map-string.
* @param mapString The string as map expression. (NotNull, NotEmpty)
* @return The map for the map-string, mutable for compatible. (NotNull, EmptyAllowed)
*/
public Map fromMapString(String mapString) {
assertStringNotNullAndNotTrimmedEmpty("mapString", mapString);
beginRemainder(mapString);
try {
assertFirstMapString();
ltrimRemainder();
removePrefixMapingBeginBrace();
final Map map = newStringObjectMap();
parseMapString(map);
assertLastRemainderAsMap(map);
return map;
} finally {
endRemainder();
}
}
// -----------------------------------------------------
// to MapString
// ------------
/**
* Convert to map object from list-string.
* @param listString The string as list expression. (NotNull, NotEmpty)
* @return The list for the list-string, mutable for compatible. (NotNull, EmptyAllowed)
*/
public List fromListString(String listString) {
assertStringNotNullAndNotTrimmedEmpty("listString", listString);
beginRemainder(listString);
try {
assertFirstListString();
ltrimRemainder();
removePrefixListingBeginBrace();
final List list = newObjectList();
parseListString(list);
assertLastRemainderAsList(list);
return list;
} finally {
endRemainder();
}
}
// ===================================================================================
// Parse
// =====
/**
* Parse the current remainder string as map.
* @param currentMap The current map made by parse process. (NotNull)
*/
protected void parseMapString(Map currentMap) {
while (true) {
if (!prepareLoopBeginning()) {
return;
}
// _/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/
// here, remainder string should starts with the key of the map
// _/_/_/_/_/_/_/_/_/_/
final int equalIndex = indexOfEqual();
assertMapStringEqualIndex(equalIndex, currentMap);
final String mapKey = substringRemainderFront(equalIndex);
removePrefixTargetIndexPlus(equalIndex, _valueEqual.length());
ltrimRemainder();
// _/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/
// here, remainder string should starts with the value of the map
// _/_/_/_/_/_/_/_/_/_/
if (startsWithMapBeginBrace()) {
removePrefixMapingBeginBrace();
parseMapString(setupNestMap(currentMap, mapKey));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
if (startsWithListBeginBrace()) {
removePrefixListingBeginBrace();
parseListString(setupNestList(currentMap, mapKey));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
final int delimiterIndex = indexOfDelimiter();
final int endBraceIndex = indexOfEndBrace();
assertMapStringEndBraceIndex(endBraceIndex, currentMap);
if (delimiterIndex >= 0 && delimiterIndex < endBraceIndex) { // delimiter exists
// e.g. value1 ; key2=value2}
final String mapValue = substringRemainderFront(delimiterIndex);
registerToMap(currentMap, filterMapKey(mapKey), filterMapListValue(mapValue));
// because the map element continues since the delimiter,
// skip the delimiter and continue the loop
removePrefixTargetIndexPlus(delimiterIndex, _elementDelimiter.length());
continue;
}
// e.g. value1} ; key2=value2}
final String mapValue = substringRemainderScope(_currentRemainderIndex, endBraceIndex);
registerToMap(currentMap, filterMapKey(mapKey), filterMapListValue(mapValue));
// analyzing map is over, so close and return.
closeByEndBraceIndex(endBraceIndex);
return;
}
}
/**
* Parse the current remainder string as list.
* @param currentList The current list made by parse process. (NotNull)
*/
protected void parseListString(List currentList) {
while (true) {
if (!prepareLoopBeginning()) {
return;
}
// _/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/_/
// here, remainder string should starts with the value of the list
// _/_/_/_/_/_/_/_/_/_/
if (startsWithMapBeginBrace()) {
removePrefixMapingBeginBrace();
parseMapString(setupNestMap(currentList));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
if (startsWithListBeginBrace()) {
removePrefixListingBeginBrace();
parseListString(setupNestList(currentList));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
final int delimiterIndex = indexOfDelimiter();
final int endBraceIndex = indexOfEndBrace();
assertListStringEndBraceIndex(endBraceIndex, currentList);
if (delimiterIndex >= 0 && delimiterIndex < endBraceIndex) { // delimiter exists
// e.g. value1 ; value2 ; value3}
final String listValue = substringRemainderFront(delimiterIndex);
currentList.add(filterMapListValue(listValue));
// because the list element continues since the delimiter,
// skip the delimiter and continue the loop.
removePrefixTargetIndexPlus(delimiterIndex, _elementDelimiter.length());
continue;
}
// e.g. value1}, value2, }
final String listValue = substringRemainderFront(endBraceIndex);
currentList.add(filterMapListValue(listValue));
// analyzing list is over, so close and return
closeByEndBraceIndex(endBraceIndex);
return;
}
}
/**
* prepare loop beginning.
* @return Does it loop next?
*/
protected boolean prepareLoopBeginning() {
removePrefixAllDelimiter();
if (isRemainderOver()) { // analyzing is over
return false;
}
if (startsWithEndBrace()) { // analyzing current map is over
removePrefixEndBrace();
return false;
}
return true;
}
/**
* Close after parse nest map list.
* @return Is is closed?
*/
protected boolean closeAfterParseNestMapList() {
if (startsWithEndBrace()) {
removePrefixEndBrace();
return true;
}
return false;
}
/**
* Close by end-brace index.
* @param endBraceIndex The index of end-brace. (NotMinus)
*/
protected void closeByEndBraceIndex(int endBraceIndex) {
_currentRemainderIndex = endBraceIndex;
removePrefixEndBrace();
}
// ===================================================================================
// Index of Control Mark
// =====================
protected int indexOfStartBrace() {
return findIndexOfControlMark(_beginBrace);
}
protected int indexOfEndBrace() {
return findIndexOfControlMark(_endBrace);
}
protected int indexOfDelimiter() {
return findIndexOfControlMark(_elementDelimiter);
}
protected int indexOfEqual() {
return findIndexOfControlMark(_valueEqual);
}
protected int findIndexOfControlMark(String controlMark) {
String current = _remainderString;
if (isEscapeCharEscape()) {
// performance headache but basically it does not come here
// because of unsupported escaping escape-char
final String escapedEscapeChar = convertToEscapedMark(_escapeChar);
current = generally_replace(current, escapedEscapeChar, convertToLengthSpace(escapedEscapeChar));
}
int currentBeginIndex = _currentRemainderIndex;
while (true) {
final int index = current.indexOf(controlMark, currentBeginIndex);
if (index < 0) { // not found
return index;
}
if (index > 0) {
final String lastChar = current.substring(index - 1, index);
if (_escapeChar.equals(lastChar)) { // escaped
final int nextIndex = index + _escapeChar.length();
currentBeginIndex = nextIndex;
continue;
}
}
return index; // found
}
}
protected String convertToLengthSpace(String value) {
final StringBuilder sb = new StringBuilder();
for (int i = 0; i < value.length(); i++) {
sb.append(" ");
}
return sb.toString();
}
// ===================================================================================
// Remove Prefix of Control Mark
// =============================
/**
* Remove prefix and all delimiters.
*/
protected void removePrefixAllDelimiter() {
ltrimRemainder();
while (true) {
if (!startsWithDelimiter()) {
break;
}
if (startsWithDelimiter()) {
removePrefixDelimiter();
ltrimRemainder();
}
}
}
/**
* Remove prefix map-prefix and begin-brace.
*/
protected void removePrefixMapingBeginBrace() {
doRemovePrefix(_mappingBeginBrace);
}
/**
* Remove prefix list-prefix and begin-brace.
*/
protected void removePrefixListingBeginBrace() {
doRemovePrefix(_listingBeginBrace);
}
/**
* Remove prefix end-brace.
*/
protected void removePrefixEndBrace() {
doRemovePrefix(_endBrace);
}
/**
* Remove prefix delimiter.
*/
protected void removePrefixDelimiter() {
doRemovePrefix(_elementDelimiter);
}
/**
* Remove prefix as mark.
* @param prefixString The string for prefix. (NotNull)
*/
protected void doRemovePrefix(String prefixString) {
assertRemovingPrefix(prefixString);
final int nonTrimmedIndex = indexOfRemainderNonTrimmedChar();
_currentRemainderIndex = nonTrimmedIndex + prefixString.length();
}
/**
* Remove prefix by the index and plus count.
* @param index The base index. (NotMinus)
* @param plusCount The plus count for index. (NotMinus)
*/
protected void removePrefixTargetIndexPlus(int index, int plusCount) {
_currentRemainderIndex = index + plusCount;
}
// ===================================================================================
// Starts/Ends with Control Mark
// =============================
/**
* Does it start with the map-prefix?
* @return The determination, true or false.
*/
protected boolean startsWithMapBeginBrace() {
final int index = indexOfRemainderNonTrimmedChar();
return index >= 0 && _remainderString.startsWith(_mappingBeginBrace, index);
}
/**
* Does it start with the list-prefix?
* @return The determination, true or false.
*/
protected boolean startsWithListBeginBrace() {
final int index = indexOfRemainderNonTrimmedChar();
return index >= 0 && _remainderString.startsWith(_listingBeginBrace, index);
}
/**
* Does it start with the delimiter?
* @return The determination, true or false.
*/
protected boolean startsWithDelimiter() {
final int index = indexOfRemainderNonTrimmedChar();
return index >= 0 && _remainderString.startsWith(_elementDelimiter, index);
}
/**
* Does it start with end-brace?
* @return The determination, true or false.
*/
protected boolean startsWithEndBrace() {
final int index = indexOfRemainderNonTrimmedChar();
return index >= 0 && _remainderString.startsWith(_endBrace, index);
}
protected boolean startsWithPrefix(String prefix) {
final int index = indexOfRemainderNonTrimmedChar();
return index >= 0 && _remainderString.startsWith(prefix, index);
}
/**
* Does it end with end-brace?
* @return The determination, true or false.
*/
protected boolean endsWithEndBrace() {
return generally_endsWith(_remainderString, _endBrace);
}
// ===================================================================================
// Map Object
// ==========
// -----------------------------------------------------
// Register
// --------
protected void registerToMap(Map currentMap, String key, Object element) {
if (isCheckDuplicateEntry()) {
assertDuplicateEntryInMap(currentMap, key, element);
}
currentMap.put(key, element);
}
protected void assertDuplicateEntryInMap(Map currentMap, String key, Object element) {
if (currentMap.containsKey(key)) {
throwDfMapDuplicateEntryException(currentMap, key, element);
}
}
protected void throwDfMapDuplicateEntryException(Map currentMap, String key, Object element) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Duplicate entry in the map string.");
br.addItem("Already Registered Entry");
br.addElement(key);
br.addElement(currentMap.get(key));
br.addItem("New Duplicate Entry");
br.addElement(key);
if (element instanceof Map, ?>) {
// might be lazy registered so might be empty (then not show)
final Map, ?> map = (Map, ?>) element;
if (!map.isEmpty()) {
br.addElement(map);
}
} else {
br.addElement(element);
}
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapDuplicateEntryException(msg);
}
// -----------------------------------------------------
// Set up
// ------
/**
* Set up new-created nest map as element of the current map.
* @param currentMap the current map to set up. (NotNull)
* @param mapKey The key of nest map. (NotNull)
* @return The new-created nest map. (NotNull)
*/
protected Map setupNestMap(Map currentMap, String mapKey) {
final Map nestMap = newStringObjectMap();
registerToMap(currentMap, filterMapKey(mapKey), nestMap);
return nestMap;
}
/**
* Set up new-created nest map as element of the current list.
* @param currentList the current list to set up. (NotNull)
* @return The new-created nest map. (NotNull)
*/
protected Map setupNestMap(List currentList) {
final Map nestMap = newStringObjectMap();
currentList.add(nestMap);
return nestMap;
}
/**
* Set up new-created nest list as element of the current map.
* @param currentMap the current map to set up. (NotNull)
* @param mapKey The key of nest map. (NotNull)
* @return The new-created nest list. (NotNull)
*/
protected List setupNestList(Map currentMap, String mapKey) {
final List nestList = newObjectList();
registerToMap(currentMap, filterMapKey(mapKey), nestList);
return nestList;
}
/**
* Set up new-created nest list as element of the current list.
* @param currentList the current map to set up. (NotNull)
* @return The new-created nest list. (NotNull)
*/
protected List setupNestList(List currentList) {
final List nestList = newObjectList();
currentList.add(nestList);
return nestList;
}
// -----------------------------------------------------
// Filter
// ------
protected String filterMapKey(String value) {
if (value == null) {
return null;
}
final String filtered = unescapeControlMark(value.trim());
return (("".equals(filtered) || "null".equals(filtered)) ? null : filtered);
}
protected String filterMapListValue(String value) {
if (value == null) {
return null;
}
final String filtered = unescapeControlMark(value.trim());
return (("".equals(filtered) || "null".equals(filtered)) ? null : filtered);
}
// -----------------------------------------------------
// Map/List Instance
// -----------------
/**
* New string-object map.
* @return The new-created map. (NotNull)
*/
public Map newStringObjectMap() { // public for client
return new LinkedHashMap();
}
/**
* New object-type list.
* @return The new-created list. (NotNull)
*/
public List newObjectList() { // public for client
return new ArrayList();
}
// ===================================================================================
// Escape Mark
// ===========
/**
* Escape control marks as plain string in the value.
* @param value The value, might contain control marks. (NullAllowed: if null, return null)
* @return The escaped string of the value. (NullAllowed: when the value is null)
*/
public String escapeControlMark(Object value) { // public for tools
if (value == null) {
return null;
}
String filtered = value.toString();
if (isEscapeCharEscape()) {
filtered = generally_replace(filtered, _escapeChar, convertToEscapedMark(_escapeChar));
}
filtered = generally_replace(filtered, _beginBrace, convertToEscapedMark(_beginBrace));
filtered = generally_replace(filtered, _endBrace, convertToEscapedMark(_endBrace));
filtered = generally_replace(filtered, _elementDelimiter, convertToEscapedMark(_elementDelimiter));
filtered = generally_replace(filtered, _valueEqual, convertToEscapedMark(_valueEqual));
return filtered;
}
/**
* Unescape control marks as plain string in the value.
* @param value The value, might contain escaped control marks. (NullAllowed: if null, return null)
* @return The unescaped string of the value. (NullAllowed: when the value is null)
*/
public String unescapeControlMark(String value) { // public for tools
if (value == null) {
return null;
}
String filtered = value;
final String escapedEscapeMark = ESCAPED_ESCAPE_MARK;
if (isEscapeCharEscape()) {
filtered = generally_replace(filtered, convertToEscapedMark(_escapeChar), escapedEscapeMark);
}
filtered = generally_replace(filtered, convertToEscapedMark(_beginBrace), _beginBrace);
filtered = generally_replace(filtered, convertToEscapedMark(_endBrace), _endBrace);
filtered = generally_replace(filtered, convertToEscapedMark(_elementDelimiter), _elementDelimiter);
filtered = generally_replace(filtered, convertToEscapedMark(_valueEqual), _valueEqual);
if (isEscapeCharEscape()) {
filtered = generally_replace(filtered, escapedEscapeMark, _escapeChar);
}
return filtered;
}
protected String convertToEscapedMark(String mark) {
return _escapeChar + mark;
}
protected boolean isEscapeCharEscape() {
// escape for escape char is unsupported (unneeded)
// so fixedly returns false
//
// compatibility is treated as important here
// o "\\n = \n" in convertValueMap.dataprop can directly work
// o plain "\" is can directly work
//
// [specification]
// escape char without control mark is treated as plain value
// e.g. "a\b\c"
//
// previous escape char of control mark is always treated as escape char
// e.g. "\;"
//
// if any spaces between the escape char and the control mark exist,
// the escape char is plain value, e.g. "\ ;"
//
return false;
}
// ===================================================================================
// Parsing Assert
// ==============
// -----------------------------------------------------
// First MapString
// ---------------
protected void assertFirstMapString() {
if (_remainderString == null) {
throwDfMapStringBasicFailureException("The map-string should not be null.");
}
if (!startsWithMapBeginBrace()) {
throwDfMapStringBasicFailureException("The map-string should start with '" + _mappingBeginBrace + "'.");
}
if (!endsWithEndBrace()) {
throwDfMapStringBasicFailureException("The map-string should end with '" + _endBrace + "'.");
}
// #for_now difficult to count non-escaped marks without performance cost (also thinking "map:" in value) by jflute (2018/05/05)
//final int beginBraceCount = countControlMarkInRemainder(_beginBrace);
//final int endBraceCount = countControlMarkInRemainder(_endBrace);
//if (beginBraceCount != endBraceCount) {
// throwDfMapStringDifferentCountBracesException(beginBraceCount, endBraceCount);
//}
}
protected void throwDfMapStringBasicFailureException(String notice) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
protected void throwDfMapStringDifferentCountBracesException(int beginCount, int endCount) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Different count between begin braces and end braces.");
br.addItem("Advice");
br.addElement("Make sure braces on your map-string.");
br.addElement("For example:");
br.addElement(" (x): map:{ sea = ");
br.addElement(" (x): map:{ sea = map:{ }");
br.addElement(" (o): map:{ sea = land }");
br.addElement(" (o): map:{ sea = map:{ land = piari } }");
br.addElement("");
br.addElement("While, map-string can escape control marks");
br.addElement("so pay attention to last char of value like this:");
br.addElement(" (x): map:{ sea = C:\\land\\piari\\} // last '}' escaped by escape char");
br.addElement(" (o): map:{ sea = C:\\land\\piari\\ } // space helps you at the case");
br.addItem("Brace Count");
br.addElement("begin: " + beginCount);
br.addElement("end: " + endCount);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// -----------------------------------------------------
// First ListString
// ----------------
protected void assertFirstListString() {
if (_remainderString == null) {
throwListStringBasicFailureException("The list string should not be null.");
}
if (!startsWithListBeginBrace()) {
throwListStringBasicFailureException("The list string should start with '" + _listingBeginBrace + "'.");
}
if (!endsWithEndBrace()) {
throwListStringBasicFailureException("The list string should end with '" + _endBrace + "'.");
}
// #for_now difficult to count non-escaped marks without performance cost (also thinking "map:" in value) by jflute (2018/05/05)
//final int startBraceCount = countControlMarkInRemainder(_beginBrace);
//final int endBraceCount = countControlMarkInRemainder(_endBrace);
//if (startBraceCount != endBraceCount) {
// throwDfListStringDifferentCountBracesException(startBraceCount, endBraceCount);
//}
}
protected void throwListStringBasicFailureException(String notice) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
protected void throwDfListStringDifferentCountBracesException(int beginCount, int endCount) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Different count between begin braces and end braces.");
br.addItem("Advice");
br.addElement("Make sure braces on your list-string.");
br.addElement("For example:");
br.addElement(" (x): list:{ sea");
br.addElement(" (x): list:{ sea ; map:{ }");
br.addElement(" (o): list:{ sea ; land }");
br.addElement(" (o): list:{ sea ; map:{ land = piari } }");
br.addElement("");
br.addElement("While, list-string can escape control marks");
br.addElement("so pay attention to last char of value like this:");
br.addElement(" (x): list:{ sea = C:\\land\\piari\\} // last '}' escaped by escape char");
br.addElement(" (o): list:{ sea = C:\\land\\piari\\ } // space helps you at the case");
br.addItem("Brace Count");
br.addElement("begin: " + beginCount);
br.addElement("end: " + endCount);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// #hope count non-escaped mark here without performance cost by jflute (2018/05/05)
//protected int countControlMarkInRemainder(String controlMark) { // only used in exception so don't mind performance
// final String remainder = dangrous_substringRemainderString(_currentRemainderIndex);
// return Srl.count(remainder, controlMark); // expects tuning of Srl in the future
//}
// -----------------------------------------------------
// ValueEqual Index
// ----------------
protected void assertMapStringEqualIndex(int valueEqualIndex, Map currentMap) {
if (_remainderString == null) { // basically no way here, just in case
final String notice = "The remainderString should not be null:";
throwDfMapValueEqualFailureException(notice, valueEqualIndex, currentMap);
}
if (valueEqualIndex < 0) {
final String notice = "Not found the value-equal mark in the map-string.";
throwDfMapValueEqualFailureException(notice, valueEqualIndex, currentMap);
}
// following checks are for internal mistake
final int logicalLength = lengthRemainder();
if (logicalLength < (_remainderString.length() - valueEqualIndex)) { // deep (or unneeded?) check (written by younger jflute)
final String notice = "The remainderString length should be greater than equalIndex:";
throwDfMapValueEqualFailureException(notice, valueEqualIndex, currentMap);
}
final String extractedMark = substringRemainderScope(valueEqualIndex, valueEqualIndex + _valueEqual.length());
if (!extractedMark.equals(_valueEqual)) { // different value-equal mark
final String notice = "The remainderString should have value-equal mark at value-equal index:";
throwDfMapValueEqualFailureException(notice, valueEqualIndex, currentMap);
}
}
protected void throwDfMapValueEqualFailureException(String notice, int valueEqualIndex, Map currentMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("ValueEqual Index");
br.addElement(valueEqualIndex);
br.addItem("Making Map");
br.addElement(currentMap);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// -----------------------------------------------------
// EndBrace Index
// --------------
protected void assertMapStringEndBraceIndex(int endBraceIndex, Map makingMap) {
if (endBraceIndex < 0) {
final String notice = "Not found the end-brace in the map-string.";
throwDfMapStringEndBraceFailureException(notice, endBraceIndex, makingMap);
}
// following checks are for internal mistake
if (_remainderString == null) {
final String notice = "The remainderString should not be null:";
throwDfMapStringEndBraceFailureException(notice, endBraceIndex, makingMap);
}
final int logicalLength = lengthRemainder();
if (logicalLength < (_remainderString.length() - endBraceIndex)) {
final String notice = "The remainderString length should be greater than end-brace index:";
throwDfMapStringEndBraceFailureException(notice, endBraceIndex, makingMap);
}
substringRemainderScope(endBraceIndex, endBraceIndex + _endBrace.length());
final String extractedMark = substringRemainderScope(endBraceIndex, endBraceIndex + _endBrace.length());
if (!extractedMark.equals(_endBrace)) {
final String notice = "The remainderString should have end-brace at the end-brace index:";
throwDfMapStringEndBraceFailureException(notice, endBraceIndex, makingMap);
}
}
protected void throwDfMapStringEndBraceFailureException(String notice, int endBraceIndex, Map currentMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("EndBrace Index");
br.addElement(endBraceIndex);
br.addItem("Making Map");
br.addElement(currentMap);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
protected void assertListStringEndBraceIndex(int endBraceIndex, List> makingList) {
if (endBraceIndex < 0) {
final String notice = "Not found the end brace.";
throwDfListStringEndBraceFailureException(notice, endBraceIndex, makingList);
}
// following checks are for internal mistake
if (_remainderString == null) {
final String notice = "The remainderString should not be null:";
throwDfListStringEndBraceFailureException(notice, endBraceIndex, makingList);
}
final int logicalLength = lengthRemainder();
if (logicalLength < (_remainderString.length() - endBraceIndex)) {
final String notice = "The remainderString length should be greater than end-brace index:";
throwDfListStringEndBraceFailureException(notice, endBraceIndex, makingList);
}
final String extractedMark = substringRemainderScope(endBraceIndex, endBraceIndex + _endBrace.length());
if (!extractedMark.equals(_endBrace)) {
final String notice = "The remainderString should have end-brace at the end-brace index:";
throwDfListStringEndBraceFailureException(notice, endBraceIndex, makingList);
}
}
protected void throwDfListStringEndBraceFailureException(String notice, int endBraceIndex, List> makingList) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("EndBrace Index");
br.addElement(endBraceIndex);
br.addItem("Making List");
br.addElement(makingList);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// -----------------------------------------------------
// Removing Prefix
// ---------------
protected void assertRemovingPrefix(String prefixString) {
if (_remainderString == null) {
final String notice = "The remainderString should not be null.";
throwDfMapRemovingPrefixFailureException(notice, prefixString);
}
if (prefixString == null) {
final String notice = "The prefixString should not be null.";
throwDfMapRemovingPrefixFailureException(notice, prefixString);
}
if (!startsWithPrefix(prefixString)) {
final String notice = "The remainderString shuold start with the prefixString.";
throwDfMapRemovingPrefixFailureException(notice, prefixString);
}
}
protected void throwDfMapRemovingPrefixFailureException(String notice, String prefixString) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Prefix String");
br.addElement(prefixString);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// -----------------------------------------------------
// Last Assertion
// --------------
protected void assertLastRemainderAsMap(Map generatedMap) {
if (indexOfRemainderNonTrimmedChar() >= 0) {
throwDfMapStringUnneededStringFoundException(generatedMap);
}
}
protected void throwDfMapStringUnneededStringFoundException(Map generatedMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("The last remainderString as map should be empty string but ...");
br.addItem("Generated Map");
br.addElement(generatedMap);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
protected void assertLastRemainderAsList(List generatedList) {
if (indexOfRemainderNonTrimmedChar() >= 0) {
throwDfListStringUnneededStringFoundException(generatedList);
}
}
protected void throwDfListStringUnneededStringFoundException(List generatedList) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("The last remainderString as list should be empty string but ...");
br.addItem("Generated List");
br.addElement(generatedList);
setupMessageRemainderString(br);
final String msg = br.buildExceptionMessage();
throw new DfMapParseFailureException(msg);
}
// -----------------------------------------------------
// Message Pattern
// ---------------
protected void setupMessageRemainderString(ExceptionMessageBuilder br) {
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("Current Remainder Index");
if (_remainderString != null) { // checking just in case
br.addElement(_currentRemainderIndex + " (max:" + _remainderString.length() + ")");
if (_currentRemainderIndex >= 0 && _remainderString.length() >= _currentRemainderIndex) { // more check
br.addElement(cut(_remainderString.substring(_currentRemainderIndex), 30, "..."));
}
} else {
br.addElement(_currentRemainderIndex);
}
br.addItem("Defined Control Mark");
br.addElement(_mapPrefix + " " + _listPrefix + " " + _beginBrace + " " + _endBrace + " " + _elementDelimiter + " " + _valueEqual);
}
protected String cut(String str, int length, String suffix) {
assertObjectNotNull("str", str);
return str.length() > length ? (str.substring(0, length) + (suffix != null ? suffix : "")) : str;
}
// ===================================================================================
// Remainder String
// ================
protected void beginRemainder(String mapString) {
_remainderString = mapString;
_currentRemainderIndex = 0;
}
protected void endRemainder() {
_remainderString = null;
_currentRemainderIndex = 0;
}
protected int indexOfRemainderNonTrimmedChar() {
return generally_indexOfNonTrimmedChar(_remainderString, _currentRemainderIndex);
}
protected boolean isRemainderOver() {
if (_remainderString.length() == _currentRemainderIndex + 1) { // may be almost here
return true;
}
if (indexOfRemainderNonTrimmedChar() < 0) { // with loop so also check just-over before this
return true;
}
return false;
}
protected int lengthRemainder() {
return _remainderString.length() - _currentRemainderIndex;
}
protected void ltrimRemainder() {
final int index = indexOfRemainderNonTrimmedChar();
if (index > 0) {
_currentRemainderIndex = index;
}
}
protected String substringRemainderFront(int endIndex) {
return _remainderString.substring(_currentRemainderIndex, endIndex);
}
protected String substringRemainderScope(int beginIndex, int endIndex) {
return _remainderString.substring(beginIndex, endIndex);
}
// ===================================================================================
// General Helper
// ==============
// no use of instance variable (can be static, but non-static for overriding)
protected String generally_replace(String str, String fromStr, String toStr) {
StringBuilder sb = null; // lazy load
int pos = 0;
int pos2 = 0;
do {
pos = str.indexOf(fromStr, pos2);
if (pos2 == 0 && pos < 0) { // first loop and not found
return str; // without creating StringBuilder
}
if (sb == null) {
sb = new StringBuilder();
}
if (pos == 0) {
sb.append(toStr);
pos2 = fromStr.length();
} else if (pos > 0) {
sb.append(str.substring(pos2, pos));
sb.append(toStr);
pos2 = pos + fromStr.length();
} else { // (pos < 0) second or after loop only
sb.append(str.substring(pos2));
return sb.toString();
}
} while (true);
}
protected boolean generally_endsWith(String str, String suffix) {
int remainderIndex = generally_lastIndexOfNonTrimmedChar(str);
for (int i = suffix.length() - 1; i >= 0; i--, remainderIndex--) {
final char branceCh = suffix.charAt(i);
final char remainderCh = str.charAt(remainderIndex);
if (branceCh != remainderCh) {
return false;
}
}
return true;
}
protected int generally_indexOfNonTrimmedChar(String str, int offsetIndex) {
for (int index = offsetIndex; index < str.length(); index++) {
final char ch = str.charAt(index);
if (ch > ' ') {
return index;
}
}
return -1;
}
protected int generally_lastIndexOfNonTrimmedChar(String str) {
for (int index = str.length() - 1; index >= 0; index--) {
final char ch = str.charAt(index);
if (ch > ' ') {
return index;
}
}
return -1;
}
protected String generally_calculateNextIndent(String previousIndent, String currentIndent) {
final StringBuilder sb = new StringBuilder();
sb.append(currentIndent);
final int indentLength = currentIndent.length() - previousIndent.length();
for (int i = 0; i < indentLength; i++) {
sb.append(" ");
}
return sb.toString();
}
protected String ln() {
return "\n";
}
protected void assertStringNotNullAndNotTrimmedEmpty(String variableName, String value) {
assertObjectNotNull("variableName", variableName);
assertObjectNotNull(variableName, value);
if (value.trim().length() == 0) {
String msg = "The value should not be empty: variableName=" + variableName + " value=" + value;
throw new IllegalArgumentException(msg);
}
}
protected void assertObjectNotNull(String variableName, Object value) {
if (variableName == null) {
throw new IllegalArgumentException("The variableName should not be null.");
}
if (value == null) {
throw new IllegalArgumentException("The argument '" + variableName + "' should not be null.");
}
}
// ===================================================================================
// Accessor
// ========
public boolean isCheckDuplicateEntry() {
return _checkDuplicateEntry;
}
public boolean isPrintOneLiner() {
return _printOneLiner;
}
}