com.liferay.portal.kernel.util.StringParser Maven / Gradle / Ivy
Show all versions of com.liferay.portal.kernel Show documentation
/**
* Copyright (c) 2000-present Liferay, Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or modify it under
* the terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at your option)
* any later version.
*
* This library is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
* FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
* details.
*/
package com.liferay.portal.kernel.util;
import com.liferay.portal.kernel.concurrent.ConcurrentReferenceValueHashMap;
import com.liferay.portal.kernel.memory.FinalizeManager;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Parses strings into parameter maps and vice versa.
*
* @author Connor McKay
* @author Brian Wing Shun Chan
* @see com.liferay.portal.kernel.portlet.Route
* @see Pattern
*/
public class StringParser {
public static StringParser create(String chunk) {
StringParser stringParser = _stringParserFragmentsCache.get(chunk);
if (stringParser == null) {
stringParser = new StringParser(chunk);
_stringParserFragmentsCache.put(chunk, stringParser);
}
return stringParser;
}
/**
* Escapes the special characters in the string so that they will have no
* special meaning in a regular expression.
*
*
* This method differs from {@link Pattern#quote(String)} by escaping each
* special character with a backslash, rather than enclosing the entire
* string in special quote tags. This allows the escaped string to be
* manipulated or have sections replaced with non-literal sequences.
*
*
* @param s the string to escape
* @return the escaped string
*/
public static String escapeRegex(String s) {
Matcher matcher = _escapeRegexPattern.matcher(s);
return matcher.replaceAll("\\\\$0");
}
/**
* Builds a string from the parameter map if this parser is appropriate.
*
*
* A parser is appropriate if each parameter matches the format of its
* accompanying fragment.
*
*
*
* If this parser is appropriate, all the parameters used in the pattern
* will be removed from the parameter map. If this parser is not
* appropriate, the parameter map will not be modified.
*
*
* @param parameters the parameter map to build the string from
* @return the string, or null
if this parser is not
* appropriate
*/
public String build(Map parameters) {
String s = _builder;
for (StringParserFragment stringParserFragment :
_stringParserFragments) {
String value = parameters.get(stringParserFragment.getName());
if (value == null) {
return null;
}
if ((_stringEncoder != null) && !stringParserFragment.isRaw()) {
value = _stringEncoder.encode(value);
}
if (!stringParserFragment.matches(value)) {
return null;
}
s = StringUtil.replace(s, stringParserFragment.getToken(), value);
}
for (StringParserFragment stringParserFragment :
_stringParserFragments) {
parameters.remove(stringParserFragment.getName());
}
return s;
}
/**
* Populates the parameter map with values parsed from the string if this
* parser matches.
*
* @param s the string to parse
* @param parameters the parameter map to populate if this parser matches
* the string
* @return true
if this parser matches; false
* otherwise
*/
public boolean parse(String s, Map parameters) {
Matcher matcher = _pattern.matcher(s);
if (!matcher.matches()) {
return false;
}
for (int i = 1; i <= _stringParserFragments.size(); i++) {
StringParserFragment stringParserFragment =
_stringParserFragments.get(i - 1);
String value = matcher.group(i);
if ((_stringEncoder != null) && !stringParserFragment.isRaw()) {
value = _stringEncoder.decode(value);
}
parameters.put(stringParserFragment.getName(), value);
}
return true;
}
/**
* Sets the string encoder to use for parsing or building a string.
*
*
* The string encoder will not be used for fragments marked as raw. A
* fragment can be marked as raw by prefixing its name with a percent sign.
*
*
* @param stringEncoder the string encoder to use for parsing or building a
* string
* @see StringEncoder
*/
public void setStringEncoder(StringEncoder stringEncoder) {
_stringEncoder = stringEncoder;
}
/**
* Constructs a new string parser from the pattern.
*
*
* The pattern can be any string containing named fragments in brackets. The
* following is a valid pattern for greeting:
*
*
*
*
*
* Hi {name}! How are you?
*
*
*
*
*
* This pattern would match the string "Hi Tom! How are you?". The
* format of a fragment may optionally be specified by inserting a colon
* followed by a regular expression after the fragment name. For instance,
* name
could be set to match only lower case letters with the
* following:
*
*
*
*
*
* Hi {name:[a-z]+}! How are you?
*
*
*
*
*
* By default, a fragment will match anything except a forward slash or a
* period.
*
*
*
* If a string parser is set to encode fragments using a {@link
* StringEncoder}, an individual fragment can be specified as raw by
* prefixing its name with a percent sign, as shown below:
*
*
*
*
*
* /view_page/{%path:.*}
*
*
*
*
*
* The format of the path fragment has also been specified to match anything
* using the pattern ".*". This pattern could be used to parse the
* string:
*
*
*
*
*
* /view_page/root/home/mysite/pages/index.htm
*
*
*
*
*
* path
would be set to
* "root/home/mysite/pages/index.htm", even if {@link
* URLStringEncoder} had been set as the string encoder.
*
*
*
* Do not include capturing subgroups in the pattern.
*
*
* @param pattern the pattern string
*/
protected StringParser(String pattern) {
String regex = escapeRegex(pattern);
Matcher matcher = _fragmentPattern.matcher(pattern);
_stringParserFragments = new ArrayList<>(matcher.groupCount());
while (matcher.find()) {
String chunk = matcher.group();
StringParserFragment stringParserFragment =
StringParserFragment.create(chunk);
_stringParserFragments.add(stringParserFragment);
pattern = StringUtil.replace(
pattern, chunk, stringParserFragment.getToken());
regex = StringUtil.replace(
regex, escapeRegex(chunk),
StringPool.OPEN_PARENTHESIS.concat(
stringParserFragment.getPattern().concat(
StringPool.CLOSE_PARENTHESIS)));
}
_builder = pattern;
_pattern = Pattern.compile(regex);
}
private static final Pattern _escapeRegexPattern = Pattern.compile(
"[\\{\\}\\(\\)\\[\\]\\*\\+\\?\\$\\^\\.\\#\\\\]");
private static final Pattern _fragmentPattern = Pattern.compile(
"\\{.+?\\}");
private static final Map _stringParserFragmentsCache =
new ConcurrentReferenceValueHashMap<>(
FinalizeManager.SOFT_REFERENCE_FACTORY);
private final String _builder;
private final Pattern _pattern;
private StringEncoder _stringEncoder;
private final List _stringParserFragments;
}