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.mapstring.MapListString Maven / Gradle / Ivy
/*
* Copyright 2014-2020 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.mapstring;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import org.dbflute.exception.MapListStringDuplicateEntryException;
import org.dbflute.exception.MapListStringParseFailureException;
import org.dbflute.helper.message.ExceptionMessageBuilder;
/**
* The string for map and list.
*
* e.g. map-string
* 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}}
*
* @author jflute
*/
public class MapListString {
// this code was written when jflute was very young
// (the code has small modifications only after created)
// ===================================================================================
// Definition
// ==========
/** The default mark of map value. */
public static final String DEFAULT_MAP_MARK = "map:";
/** The default mark of list value. */
public static final String DEFAULT_LIST_MARK = "list:";
/** The default control mark of start-brace. */
public static final String DEFAULT_START_BRACE = "{";
/** The default control mark of end-brace. */
public static final String DEFAULT_END_BRACE = "}";
/** The default control mark of delimiter. */
public static final String DEFAULT_DELIMITER = ";";
/** The default control mark of equal. */
public static final String DEFAULT_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
// =========
// -----------------------------------------------------
// Component
// ---------
/** The mark of map. (NotNull: but changeable) */
protected String _mapMark;
/** The mark of list. (NotNull: but changeable) */
protected String _listMark;
/** The control mark of start brace. (NotNull: but changeable) */
protected String _startBrace;
/** The control mark of end brace. (NotNull: but changeable) */
protected String _endBrace;
/** The control mark of delimiter. (NotNull: but changeable) */
protected String _delimiter;
/** The control mark of equal for map-string. (NotNull: but changeable) */
protected String _equal;
/** The escape character for control marks. (NotNull) */
protected final String _escapeChar;
// -----------------------------------------------------
// Temporary
// ---------
/** The string of top (full) string as temporary variable for generation. (NullAllowed: depends on process) */
protected String _topString;
/** The string of remainder as temporary variable for generation. (NullAllowed: depends on process) */
protected String _remainderString;
// -----------------------------------------------------
// 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 setting as default.
* You can change marks by setters after creation.
*/
public MapListString() {
_mapMark = DEFAULT_MAP_MARK;
_listMark = DEFAULT_LIST_MARK;
_startBrace = DEFAULT_START_BRACE;
_endBrace = DEFAULT_END_BRACE;
_delimiter = DEFAULT_DELIMITER;
_equal = DEFAULT_EQUAL;
_escapeChar = ESCAPE_CHAR; // fixed for now
}
// ===================================================================================
// Build String
// ============
/**
* Build map-string from the map object.
* @param map The map object that has string keys. (NotNull)
* @return The string as map expression. (NotNull)
*/
public String buildMapString(Map map) {
final StringBuilder sb = new StringBuilder();
@SuppressWarnings("unchecked")
final Map casted = (Map) map;
final boolean printOneLiner = isPrintOneLiner();
doBuildMapString(sb, casted, printOneLiner, "", printOneLiner ? "" : " ");
return sb.toString();
}
protected void doBuildMapString(StringBuilder sb, Map map, boolean printOneLiner, String preIndent, String curIndent) {
sb.append(_mapMark).append(_startBrace);
if (!map.isEmpty()) {
int index = 0;
for (Entry entry : map.entrySet()) {
final String key = entry.getKey();
final Object value = entry.getValue();
if (printOneLiner) {
if (index > 0) {
sb.append(" ").append(_delimiter);
}
} else {
sb.append(ln()).append(curIndent).append(_delimiter);
}
sb.append(" ").append(escapeControlMark(key)).append(" ").append(_equal).append(" ");
if (value instanceof Map, ?>) {
@SuppressWarnings("unchecked")
final Map valueMap = (Map) value;
final String nextIndent = printOneLiner ? "" : calculateNextIndent(preIndent, curIndent);
doBuildMapString(sb, valueMap, printOneLiner, curIndent, nextIndent);
} else if (value instanceof List>) {
@SuppressWarnings("unchecked")
final List valueList = (List) value;
final String nextIndent = printOneLiner ? "" : calculateNextIndent(preIndent, curIndent);
doBuildListString(sb, valueList, printOneLiner, curIndent, nextIndent);
} else {
sb.append(escapeControlMark(value));
}
++index;
}
if (printOneLiner) {
sb.append(" ");
} else {
sb.append(ln()).append(preIndent);
}
}
sb.append(_endBrace);
}
/**
* Build list-string from the list object.
* @param list The list object that has object elements. (NotNull)
* @return The string as list expression. (NotNull)
*/
public String buildListString(List extends Object> list) {
final StringBuilder sb = new StringBuilder();
@SuppressWarnings("unchecked")
final List casted = (List) list;
final boolean printOneLiner = isPrintOneLiner();
doBuildListString(sb, casted, printOneLiner, "", printOneLiner ? "" : " ");
return sb.toString();
}
protected void doBuildListString(StringBuilder sb, List extends Object> list, boolean printOneLiner, String preIndent,
String curIndent) {
sb.append(_listMark).append(_startBrace);
if (!list.isEmpty()) {
int index = 0;
for (Object value : list) {
if (printOneLiner) {
if (index > 0) {
sb.append(" ").append(_delimiter);
}
} else {
sb.append(ln()).append(curIndent).append(_delimiter);
}
sb.append(" ");
if (value instanceof Map, ?>) {
@SuppressWarnings("unchecked")
final Map valueMap = (Map) value;
final String nextIndent = printOneLiner ? "" : calculateNextIndent(preIndent, curIndent);
doBuildMapString(sb, valueMap, printOneLiner, curIndent, nextIndent);
} else if (value instanceof List>) {
@SuppressWarnings("unchecked")
final List valueList = (List) value;
final String nextIndent = printOneLiner ? "" : calculateNextIndent(preIndent, curIndent);
doBuildListString(sb, valueList, printOneLiner, curIndent, nextIndent);
} else {
sb.append(escapeControlMark(value));
}
++index;
}
if (printOneLiner) {
sb.append(" ");
} else {
sb.append(ln()).append(preIndent);
}
}
sb.append(_endBrace);
}
protected String calculateNextIndent(String preIndent, String curIndent) {
final StringBuilder sb = new StringBuilder();
sb.append(curIndent);
final int indentLength = curIndent.length() - preIndent.length();
for (int i = 0; i < indentLength; i++) {
sb.append(" ");
}
return sb.toString();
}
// ===================================================================================
// Generate Object
// ===============
/**
* Generate map object from the map-string.
* @param mapString The string as map expression. (NotNull)
* @return The generated map. (NotNull)
*/
public Map generateMap(String mapString) {
assertMapString(mapString);
_topString = mapString;
_remainderString = mapString;
removeBothSideSpaceAndTabAndNewLine();
removePrefixMapMarkAndStartBrace();
final Map generatedMap = newStringObjectMap();
parseRemainderMapString(generatedMap);
if (!"".equals(_remainderString)) {
throwMapStringUnneededStringFoundException(mapString, generatedMap);
}
return generatedMap;
}
protected void throwMapStringUnneededStringFoundException(String mapString, Map generatedMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Final remainderString should be empty string but ...");
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("Map String");
br.addElement(mapString);
br.addItem("Generated Map");
br.addElement(generatedMap);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
/**
* Generate map object from list-string.
* @param listString The string as list expression. (NotNull)
* @return The generated list. (NotNull)
*/
public List generateList(String listString) {
assertListString(listString);
_topString = listString;
_remainderString = listString;
removeBothSideSpaceAndTabAndNewLine();
removePrefixListMarkAndStartBrace();
final List generatedList = newObjectList();
parseRemainderListString(generatedList);
if (!"".equals(_remainderString)) {
throwListStringUnneededStringFoundException(listString, generatedList);
}
return generatedList;
}
protected void throwListStringUnneededStringFoundException(String listString, List generatedList) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Final remainderString should be empty string but ...");
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("List String");
br.addElement(listString);
br.addItem("Generated List");
br.addElement(generatedList);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
// ===================================================================================
// Parse
// =====
/**
* Parse the current remainder string as map.
* @param currentMap The current map made by parse process. (NotNull)
*/
protected void parseRemainderMapString(final Map currentMap) {
while (true) {
if (initializeAtLoopBeginning()) {
return;
}
// *** now, _remainderString should starts with the key of the map ***
final int equalIndex = indexOfEqual();
assertMapStringEqualIndex(_remainderString, equalIndex, _topString, currentMap);
final String mapKey = _remainderString.substring(0, equalIndex).trim();
removePrefixTargetIndexPlus(equalIndex, _equal.length());
removeBothSideSpaceAndTabAndNewLine();
// *** now, _remainderString should starts with the value of the map ***
if (isStartsWithMapPrefix(_remainderString)) {
removePrefixMapMarkAndStartBrace();
parseRemainderMapString(setupNestMap(currentMap, mapKey));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
if (isStartsWithListPrefix(_remainderString)) {
removePrefixListMarkAndStartBrace();
parseRemainderListString(setupNestList(currentMap, mapKey));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
final int delimiterIndex = indexOfDelimiter();
final int endBraceIndex = indexOfEndBrace();
assertMapStringEndBraceIndex(_remainderString, endBraceIndex, _topString, currentMap);
if (delimiterIndex >= 0 && delimiterIndex < endBraceIndex) { // delimiter exists
// e.g. value1 ; key2=value2}
final String mapValue = _remainderString.substring(0, delimiterIndex);
registerToMap(currentMap, filterMapListKey(mapKey), filterMapListValue(mapValue));
// because the map element continues since the delimiter,
// skip the delimiter and continue the loop
removePrefixTargetIndexPlus(delimiterIndex, _delimiter.length());
continue;
}
// e.g. value1} ; key2=value2}
final String mapValue = _remainderString.substring(0, endBraceIndex);
registerToMap(currentMap, filterMapListKey(mapKey), filterMapListValue(mapValue));
// analyzing map is over, so close and return.
closeByEndBraceIndex(endBraceIndex);
return;
}
}
/**
* Parse remainder list string.
* @param currentList current list.
*/
protected void parseRemainderListString(final List currentList) {
while (true) {
if (initializeAtLoopBeginning()) {
return;
}
// *** now, _remainderString should starts with the value of the list ***
if (isStartsWithMapPrefix(_remainderString)) {
removePrefixMapMarkAndStartBrace();
parseRemainderMapString(setupNestMap(currentList));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
if (isStartsWithListPrefix(_remainderString)) {
removePrefixListMarkAndStartBrace();
parseRemainderListString(setupNestList(currentList));
if (closeAfterParseNestMapList()) {
return;
}
continue;
}
final int delimiterIndex = indexOfDelimiter();
final int endBraceIndex = indexOfEndBrace();
assertListStringEndBraceIndex(_remainderString, endBraceIndex, _topString, currentList);
if (delimiterIndex >= 0 && delimiterIndex < endBraceIndex) { // delimiter exists
// e.g. value1 ; value2 ; value3}
final String listValue = _remainderString.substring(0, delimiterIndex);
currentList.add(filterMapListValue(listValue));
// because the list element continues since the delimiter,
// skip the delimiter and continue the loop.
removePrefixTargetIndexPlus(delimiterIndex, _delimiter.length());
continue;
}
// e.g. value1}, value2, }
final String listValue = _remainderString.substring(0, endBraceIndex);
currentList.add(filterMapListValue(listValue));
// analyzing list is over, so close and return
closeByEndBraceIndex(endBraceIndex);
return;
}
}
/**
* Initialize at loop beginning.
* @return Is is end?
*/
protected boolean initializeAtLoopBeginning() {
// remove prefix delimiter (result string is always trimmed)
removePrefixAllDelimiter();
if (_remainderString.equals("")) { // analyzing is over
return true;
}
if (isStartsWithEndBrace(_remainderString)) { // analyzing current map is over
removePrefixEndBrace();
return true;
}
return false;
}
/**
* Close after parse nest map list.
* @return Is is closed?
*/
protected boolean closeAfterParseNestMapList() {
if (isStartsWithEndBrace(_remainderString)) {
removePrefixEndBrace();
return true;
}
return false;
}
/**
* Close by end-brace index.
* @param endBraceIndex The index of end-brace. (NotMinus)
*/
protected void closeByEndBraceIndex(int endBraceIndex) {
_remainderString = _remainderString.substring(endBraceIndex);
removePrefixEndBrace();
}
protected int indexOfStartBrace() {
return findIndexOfControlMark(_remainderString, _startBrace);
}
protected int indexOfEndBrace() {
return findIndexOfControlMark(_remainderString, _endBrace);
}
protected int indexOfDelimiter() {
return findIndexOfControlMark(_remainderString, _delimiter);
}
protected int indexOfEqual() {
return findIndexOfControlMark(_remainderString, _equal);
}
protected int findIndexOfControlMark(String remainderString, String controlMark) {
String current = remainderString;
if (isEscapeCharEscape()) {
final String escapedEscapeChar = toEscapedMark(_escapeChar);
current = replace(current, escapedEscapeChar, buildLengthSpace(escapedEscapeChar));
}
int baseIndex = 0;
while (true) {
final int index = current.indexOf(controlMark);
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();
baseIndex = baseIndex + nextIndex;
current = current.substring(nextIndex);
continue;
}
}
return baseIndex + index; // found
}
}
protected String buildLengthSpace(String value) {
final StringBuilder sb = new StringBuilder();
for (int i = 0; i < value.length(); i++) {
sb.append(" ");
}
return sb.toString();
}
// ===================================================================================
// Remove
// ======
/**
* Remove prefix map-mark and start-brace.
*/
protected void removePrefixMapMarkAndStartBrace() {
removePrefix(_mapMark + _startBrace);
}
/**
* Remove prefix list-mark and start-brace.
*/
protected void removePrefixListMarkAndStartBrace() {
removePrefix(_listMark + _startBrace);
}
/**
* Remove prefix delimiter.
*/
protected void removePrefixDelimiter() {
removePrefix(_delimiter);
}
/**
* Remove prefix end-brace.
*/
protected void removePrefixEndBrace() {
removePrefix(_endBrace);
}
/**
* Remove prefix as mark.
* @param prefixString The string for prefix. (NotNull)
*/
protected void removePrefix(String prefixString) {
if (_remainderString == null) {
final String notice = "The remainderString should not be null.";
throwMapListStringPrefixFailureException(notice, prefixString, _topString);
}
if (prefixString == null) {
final String notice = "The prefixString should not be null.";
throwMapListStringPrefixFailureException(notice, prefixString, _topString);
}
removeBothSideSpaceAndTabAndNewLine();
// deep (or unneeded?) check
if (_remainderString.length() < prefixString.length()) {
final String notice = "The remainderString length shuold be greater than the prefixString length.";
throwMapListStringPrefixFailureException(notice, prefixString, _topString);
}
if (!_remainderString.startsWith(prefixString)) {
final String notice = "The remainderString shuold start with the prefixString.";
throwMapListStringPrefixFailureException(notice, prefixString, _topString);
}
_remainderString = _remainderString.substring(prefixString.length());
removeBothSideSpaceAndTabAndNewLine();
}
protected void throwMapListStringPrefixFailureException(String notice, String prefixString, String mapString) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("Prefix String");
br.addElement(prefixString);
br.addItem("MapList String");
br.addElement(mapString);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
/**
* Remove prefix and all delimiters.
*/
protected void removePrefixAllDelimiter() {
removeBothSideSpaceAndTabAndNewLine();
while (true) {
if (!isStartsWithDelimiter(_remainderString)) {
break;
}
if (isStartsWithDelimiter(_remainderString)) {
removePrefixDelimiter();
removeBothSideSpaceAndTabAndNewLine();
}
}
}
/**
* Remove both side space and tab and new-line.
*/
protected void removeBothSideSpaceAndTabAndNewLine() {
_remainderString = _remainderString.trim();
}
/**
* 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) {
_remainderString = _remainderString.substring(index + plusCount);
}
// ===================================================================================
// Filter
// ======
protected String filterMapListKey(String value) {
if (value == null) {
return null;
}
value = value.trim();
value = unescapeControlMark(value);
return (("".equals(value) || "null".equals(value)) ? null : value);
}
protected String filterMapListValue(String value) {
if (value == null) {
return null;
}
value = value.trim();
value = unescapeControlMark(value);
return (("".equals(value) || "null".equals(value)) ? null : value);
}
// ===================================================================================
// Determination
// =============
/**
* Does it start with the map-prefix?
* @param targetString The target string to determine. (NotNull)
* @return The determination, true or false.
*/
protected boolean isStartsWithMapPrefix(String targetString) {
if (targetString == null) {
String msg = "The argument 'targetString' should not be null.";
throw new IllegalArgumentException(msg);
}
targetString = targetString.trim();
return targetString.startsWith(getMapPrefix());
}
protected String getMapPrefix() {
return _mapMark + _startBrace;
}
/**
* Does it start with the list-prefix?
* @param targetString The target-string to determine. (NotNull)
* @return The determination, true or false.
*/
protected boolean isStartsWithListPrefix(String targetString) {
if (targetString == null) {
String msg = "The argument 'targetString' should not be null.";
throw new IllegalArgumentException(msg);
}
targetString = targetString.trim();
return targetString.startsWith(_listMark + _startBrace);
}
protected String getListPrefix() {
return _listMark + _startBrace;
}
/**
* Does it start with the delimiter?
* @param targetString The target string to determine. (NotNull)
* @return The determination, true or false.
*/
protected boolean isStartsWithDelimiter(String targetString) {
if (targetString == null) {
String msg = "The argument 'targetString' should not be null.";
throw new IllegalArgumentException(msg);
}
targetString = targetString.trim();
return targetString.startsWith(_delimiter);
}
/**
* Does it start with end-brace?
* @param targetString The target string to determine. (NotNull)
* @return The determination, true or false.
*/
protected boolean isStartsWithEndBrace(String targetString) {
if (targetString == null) {
String msg = "The argument 'targetString' should not be null.";
throw new IllegalArgumentException(msg);
}
targetString = targetString.trim();
return targetString.startsWith(_endBrace);
}
/**
* Does it end with end-brace?
* @param targetString The target string to determine. (NotNull)
* @return The determination, true or false.
*/
protected boolean isEndsWithEndBrace(String targetString) {
if (targetString == null) {
String msg = "The argument 'targetString' should not be null.";
throw new IllegalArgumentException(msg);
}
targetString = targetString.trim();
return targetString.endsWith(_endBrace);
}
// ===================================================================================
// Setup MapList
// =============
/**
* 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, filterMapListKey(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, filterMapListKey(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;
}
/**
* New string-object map.
* @return The new-created map. (NotNull)
*/
protected Map newStringObjectMap() {
return new LinkedHashMap();
}
/**
* New object-type list.
* @return The new-created list. (NotNull)
*/
protected List newObjectList() {
return new ArrayList();
}
// ===================================================================================
// Map Registration
// ================
protected void registerToMap(Map currentMap, String key, Object element) {
doCheckDuplicateEntryIfNeeds(currentMap, key, element);
currentMap.put(key, element);
}
protected void doCheckDuplicateEntryIfNeeds(Map currentMap, String key, Object element) {
if (isCheckDuplicateEntry()) {
if (currentMap.containsKey(key)) {
throwMapStringDuplicateEntryFoundException(currentMap, key, element);
}
}
}
protected void throwMapStringDuplicateEntryFoundException(Map currentMap, String key, Object element) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Duplicate entry in the map string.");
br.addItem("MapList String");
br.addElement(_topString);
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);
}
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringDuplicateEntryException(msg);
}
// ===================================================================================
// Escape
// ======
/**
* 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 = replace(filtered, _escapeChar, toEscapedMark(_escapeChar));
}
filtered = replace(filtered, _startBrace, toEscapedMark(_startBrace));
filtered = replace(filtered, _endBrace, toEscapedMark(_endBrace));
filtered = replace(filtered, _delimiter, toEscapedMark(_delimiter));
filtered = replace(filtered, _equal, toEscapedMark(_equal));
return filtered;
}
/**
* Un-escape control marks as plain string in the value.
* @param value The value, might contain escaped control marks. (NullAllowed: if null, return null)
* @return The un-escaped 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 = replace(filtered, toEscapedMark(_escapeChar), escapedEscapeMark);
}
filtered = replace(filtered, toEscapedMark(_startBrace), _startBrace);
filtered = replace(filtered, toEscapedMark(_endBrace), _endBrace);
filtered = replace(filtered, toEscapedMark(_delimiter), _delimiter);
filtered = replace(filtered, toEscapedMark(_equal), _equal);
if (isEscapeCharEscape()) {
filtered = replace(filtered, escapedEscapeMark, _escapeChar);
}
return filtered;
}
protected String toEscapedMark(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.dfprop 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;
}
// ===================================================================================
// Assert Helper
// =============
protected void assertMapString(String mapString) {
if (mapString == null) {
final String notice = "The map string should not be null.";
throwMapStringBasicFailureException(notice, mapString);
}
mapString = mapString.trim();
if (!isStartsWithMapPrefix(mapString)) {
final String notice = "The map string should start with '" + getMapPrefix() + "'.";
throwMapStringBasicFailureException(notice, mapString);
}
if (!isEndsWithEndBrace(mapString)) {
final String notice = "The map string should end with '" + _endBrace + "'.";
throwMapStringBasicFailureException(notice, mapString);
}
final int startBraceCount = getControlMarkCount(mapString, _startBrace);
final int endBraceCount = getControlMarkCount(mapString, _endBrace);
if (startBraceCount != endBraceCount) {
throwMapStringDifferentCountBracesException(mapString, startBraceCount, endBraceCount);
}
}
protected void throwMapStringBasicFailureException(String notice, String mapString) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("Map String");
br.addElement(mapString);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void throwMapStringDifferentCountBracesException(String mapString, int startCount, int endCount) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Different count between start braces and end braces.");
br.addItem("Advice");
br.addElement("Make sure braces on your map-list string.");
br.addElement("For example:");
br.addElement(" (o): map:{ foo = bar }");
br.addElement(" (o): map:{ foo = map:{ bar = qux } }");
br.addElement(" (x): map:{ foo = ");
br.addElement(" (x): map:{ foo = map:{ }");
br.addElement("");
br.addElement("map-list string can escape control marks");
br.addElement("so pay attention to last char of value like this:");
br.addElement(" (x): map:{ foo = C:\\foo\\bar\\} // last '}' escaped by escape char");
br.addElement(" (o): map:{ foo = C:\\foo\\bar\\ } // space helps you at the case");
br.addItem("Map String");
br.addElement(mapString);
br.addItem("Brace Count");
br.addElement("start: " + startCount);
br.addElement("end: " + endCount);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void assertListString(String listString) {
if (listString == null) {
final String notice = "The list string should not be null.";
throwListStringBasicFailureException(notice, listString);
}
listString = listString.trim();
if (!isStartsWithListPrefix(listString)) {
final String notice = "The list string should start with '" + getListPrefix() + "'.";
throwListStringBasicFailureException(notice, listString);
}
if (!isEndsWithEndBrace(listString)) {
final String notice = "The list string should end with '" + _endBrace + "'.";
throwListStringBasicFailureException(notice, listString);
}
final int startBraceCount = getControlMarkCount(listString, _startBrace);
final int endBraceCount = getControlMarkCount(listString, _endBrace);
if (startBraceCount != endBraceCount) {
throwListStringDifferentCountBracesException(listString, startBraceCount, endBraceCount);
}
}
protected void throwListStringBasicFailureException(String notice, String listString) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder String");
br.addElement(_remainderString);
br.addItem("List String");
br.addElement(listString);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void throwListStringDifferentCountBracesException(String listString, int startCount, int endCount) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice("Different count between start braces and end braces.");
br.addItem("List String");
br.addElement(listString);
br.addItem("Brace Count");
br.addElement("Start: " + startCount);
br.addElement("End: " + endCount);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected int getControlMarkCount(String targetString, String controlMark) {
int result = 0;
String current = targetString;
while (true) {
final int index = findIndexOfControlMark(current, controlMark);
if (index < 0) {
break;
}
result++;
current = current.substring(index + controlMark.length());
}
if (result == 0) {
result = -1;
}
return result;
}
protected void assertMapStringEqualIndex(String remainderString, int equalIndex, String mapString, Map currentMap) {
if (remainderString == null) {
final String notice = "The remainderString should not be null:";
throwMapStringEqualFailureException(notice, remainderString, equalIndex, mapString, currentMap);
}
if (equalIndex < 0) {
final String notice = "Not found the equal mark in the map.";
throwMapStringEqualFailureException(notice, remainderString, equalIndex, mapString, currentMap);
}
// deep (or unneeded?) check (written by younger jflute)
if (remainderString.length() < equalIndex) {
final String notice = "The remainderString length should be greater than equalIndex:";
throwMapStringEqualFailureException(notice, remainderString, equalIndex, mapString, currentMap);
}
final String extractedMark = remainderString.substring(equalIndex, equalIndex + _equal.length());
if (!extractedMark.equals(_equal)) {
final String notice = "The remainderString should have equal mark at equalIndex:";
throwMapStringEqualFailureException(notice, remainderString, equalIndex, mapString, currentMap);
}
}
protected void throwMapStringEqualFailureException(String notice, String remainderMapString, int equalIndex, String mapString,
Map currentMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder String");
br.addElement(remainderMapString);
br.addItem("Equal Index");
br.addElement(equalIndex);
br.addItem("Whole Map String");
br.addElement(mapString);
br.addItem("Making Map");
br.addElement(currentMap);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void assertMapStringEndBraceIndex(String remainderString, int endBraceIndex, String mapString,
Map currentMap) {
if (remainderString == null) {
final String notice = "The remainderString should not be null:";
throwMapStringEndBraceFailureException(notice, remainderString, endBraceIndex, mapString, currentMap);
}
if (endBraceIndex < 0) {
final String notice = "Not found the end brace.";
throwMapStringEndBraceFailureException(notice, remainderString, endBraceIndex, mapString, currentMap);
}
// deep (or unneeded?) check (written by younger jflute)
if (remainderString.length() < endBraceIndex) {
final String notice = "The remainderString length should be greater than endMarkIndex:";
throwMapStringEndBraceFailureException(notice, remainderString, endBraceIndex, mapString, currentMap);
}
final String extractedMark = remainderString.substring(endBraceIndex, endBraceIndex + _endBrace.length());
if (!extractedMark.equals(_endBrace)) {
final String notice = "The remainderString should have end brace at the endMarkIndex:";
throwMapStringEndBraceFailureException(notice, remainderString, endBraceIndex, mapString, currentMap);
}
}
protected void assertListStringEndBraceIndex(String remainderString, int endBraceIndex, String listString, List> currentList) {
if (remainderString == null) {
final String notice = "The remainderString should not be null:";
throwListStringEndBraceFailureException(notice, remainderString, endBraceIndex, listString, currentList);
}
if (endBraceIndex < 0) {
final String notice = "Not found the end brace.";
throwListStringEndBraceFailureException(notice, remainderString, endBraceIndex, listString, currentList);
}
// deep (or unneeded?) check (written by younger jflute)
if (remainderString.length() < endBraceIndex) {
final String notice = "The remainderString length should be greater than endMarkIndex:";
throwListStringEndBraceFailureException(notice, remainderString, endBraceIndex, listString, currentList);
}
final String extractedMark = remainderString.substring(endBraceIndex, endBraceIndex + _endBrace.length());
if (!extractedMark.equals(_endBrace)) {
final String notice = "The remainderString should have end brace at the endMarkIndex:";
throwListStringEndBraceFailureException(notice, remainderString, endBraceIndex, listString, currentList);
}
}
protected void throwMapStringEndBraceFailureException(String notice, String remainderMapString, int equalIndex, String mapString,
Map currentMap) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder Map String");
br.addElement(remainderMapString);
br.addItem("EndBrace Index");
br.addElement(equalIndex);
br.addItem("Whole Map String");
br.addElement(mapString);
br.addItem("Making Map");
br.addElement(currentMap);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void throwListStringEndBraceFailureException(String notice, String remainderMapString, int equalIndex, String listString,
List> currentList) {
final ExceptionMessageBuilder br = new ExceptionMessageBuilder();
br.addNotice(notice);
br.addItem("Remainder List String");
br.addElement(remainderMapString);
br.addItem("EndBrace Index");
br.addElement(equalIndex);
br.addItem("Whole List String");
br.addElement(listString);
br.addItem("Making List");
br.addElement(currentList);
prepareControlMarkMessage(br);
final String msg = br.buildExceptionMessage();
throw new MapListStringParseFailureException(msg);
}
protected void prepareControlMarkMessage(final ExceptionMessageBuilder br) {
br.addItem("Control Marks");
br.addElement(_startBrace + " " + _endBrace + " " + _delimiter + " " + _equal);
}
// ===================================================================================
// General Helper
// ==============
protected String 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 String ln() {
return "\n";
}
// ===================================================================================
// Accessor
// ========
public void setMapMark(String mapMark) {
_mapMark = mapMark;
}
public void setListMark(String listMark) {
_listMark = listMark;
}
public void setStartBrace(String startBrace) {
_startBrace = startBrace;
}
public void setEndBrace(String endBrace) {
_endBrace = endBrace;
}
public void setDelimiter(String delimiter) {
_delimiter = delimiter;
}
public void setEqual(String equal) {
_equal = equal;
}
// -----------------------------------------------------
// Option
// ------
/**
* Check duplicate entry of map for generation. (throws exception if found)
* @return this;
*/
public MapListString checkDuplicateEntry() {
_checkDuplicateEntry = true;
return this;
}
public boolean isCheckDuplicateEntry() {
return _checkDuplicateEntry;
}
/**
* Print as one liner for building.
* @return this;
*/
public MapListString printOneLiner() {
_printOneLiner = true;
return this;
}
public boolean isPrintOneLiner() {
return _printOneLiner;
}
}