All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.liferay.portal.kernel.util.StringParser Maven / Gradle / Ivy

Go to download

Contains interfaces for the portal services. Interfaces are only loaded by the global class loader and are shared by all plugins.

There is a newer version: 156.0.0
Show newest version
/**
 * 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; }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy