Account GET data
authorGustavo Martin Morcuende <gu.martinm@gmail.com>
Mon, 3 Oct 2016 20:13:09 +0000 (22:13 +0200)
committerGustavo Martin Morcuende <gu.martinm@gmail.com>
Mon, 3 Oct 2016 20:13:09 +0000 (22:13 +0200)
src/main/java/com/prueba/authorization/persistence/dao/ApplicationResourceDao.java [new file with mode: 0644]
src/main/java/com/prueba/authorization/persistence/dao/AuthorizationDao.java [deleted file]
src/main/java/com/prueba/authorization/services/impl/AuthorizationServicesImpl.java
src/main/java/com/prueba/core/context/util/AntPathMatcher.java [new file with mode: 0644]
src/main/java/com/prueba/core/context/util/CollectionUtils.java [new file with mode: 0644]
src/main/java/com/prueba/core/context/util/ObjectUtils.java [new file with mode: 0644]
src/main/java/com/prueba/core/context/util/StringUtils.java [new file with mode: 0644]
src/main/java/com/prueba/persistence/dao/AccountDao.java
src/main/java/com/prueba/resources/controllers/ApiController.java
src/main/java/com/prueba/services/impl/LoginServiceImpl.java

diff --git a/src/main/java/com/prueba/authorization/persistence/dao/ApplicationResourceDao.java b/src/main/java/com/prueba/authorization/persistence/dao/ApplicationResourceDao.java
new file mode 100644 (file)
index 0000000..c8dbe70
--- /dev/null
@@ -0,0 +1,46 @@
+package com.prueba.authorization.persistence.dao;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import javax.sql.DataSource;
+
+import com.prueba.core.context.integration.database.impl.DataBaseAccessImpl;
+import com.prueba.core.context.web.application.ApplicationWebContext;
+
+public class ApplicationResourceDao {
+       public static final String URL_PATTERN = "URL_PATTERN";
+       public static final String HTTP_METHOD = "HTTP_METHOD";
+
+       public List<Map<String, String>> findURLsByUserName(String userName) {
+               final DataSource dataSource = ApplicationWebContext.getInstance().getDataSource();
+               final DataBaseAccessImpl dataBaseAccess = new DataBaseAccessImpl(dataSource);
+               
+               return dataBaseAccess.executeQuery(""
+                               + "SELECT APP_RES.URL_PATTERN, APP_RES.HTTP_METHOD FROM APPLICATION_ROLE APP_ROLE "
+                               + "INNER JOIN APPLICATION_RESOURCE_APPLICATION_ROLE APP_RES_APP_ROLE ON APP_ROLE.CODE = APP_RES_APP_ROLE.APPLICATION_ROLE_CODE "
+                               + "INNER JOIN APPLICATION_RESOURCE APP_RES ON APP_RES.URL_PATTERN = APP_RES_APP_ROLE.APPLICATION_RESOURCE_URL_PATTERN "
+                               + "INNER JOIN ACCOUNT ACC ON ACC.APPLICATION_ROLE_CODE = APP_ROLE.CODE "
+                               + "WHERE ACC.CODE = ? ",
+                               answer ->
+               {
+                       final List<Map<String, String>> result = new ArrayList<>();
+                       while (answer.next()) {
+                               final Map<String, String> row = new HashMap<>();
+                               String urlPatternValue = answer.getString(URL_PATTERN);
+                               String httpMethodValue = answer.getString(HTTP_METHOD);
+                               row.put(URL_PATTERN, urlPatternValue);
+                               row.put(HTTP_METHOD, httpMethodValue);
+                               result.add(row);
+                       }
+              
+                       return result;
+               },
+               preparedStatement -> {
+                       preparedStatement.setString(1, userName);
+               });
+       }
+}
+
diff --git a/src/main/java/com/prueba/authorization/persistence/dao/AuthorizationDao.java b/src/main/java/com/prueba/authorization/persistence/dao/AuthorizationDao.java
deleted file mode 100644 (file)
index 947c206..0000000
+++ /dev/null
@@ -1,46 +0,0 @@
-package com.prueba.authorization.persistence.dao;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-import java.util.List;
-import java.util.Map;
-
-import javax.sql.DataSource;
-
-import com.prueba.core.context.integration.database.impl.DataBaseAccessImpl;
-import com.prueba.core.context.web.application.ApplicationWebContext;
-
-public class AuthorizationDao {
-       public static final String URL_PATTERN = "URL_PATTERN";
-       public static final String HTTP_METHOD = "HTTP_METHOD";
-
-       public List<Map<String, String>> findURLsByUserName(String userName) {
-               final DataSource dataSource = ApplicationWebContext.getInstance().getDataSource();
-               final DataBaseAccessImpl dataBaseAccess = new DataBaseAccessImpl(dataSource);
-               
-               return dataBaseAccess.executeQuery(""
-                               + "SELECT APP_RES.URL_PATTERN, APP_RES.HTTP_METHOD FROM APPLICATION_ROLE APP_ROLE "
-                               + "INNER JOIN APPLICATION_RESOURCE_APPLICATION_ROLE APP_RES_APP_ROLE ON APP_ROLE.CODE = APP_RES_APP_ROLE.APPLICATION_ROLE_CODE "
-                               + "INNER JOIN APPLICATION_RESOURCE APP_RES ON APP_RES.URL_PATTERN = APP_RES_APP_ROLE.APPLICATION_RESOURCE_URL_PATTERN "
-                               + "INNER JOIN ACCOUNT ACC ON ACC.APPLICATION_ROLE_CODE = APP_ROLE.CODE "
-                               + "WHERE ACC.CODE = ? ",
-                               answer ->
-               {
-                       final List<Map<String, String>> result = new ArrayList<>();
-                       while (answer.next()) {
-                               final Map<String, String> row = new HashMap<>();
-                               String urlPatternValue = answer.getString(URL_PATTERN);
-                               String httpMethodValue = answer.getString(HTTP_METHOD);
-                               row.put(URL_PATTERN, urlPatternValue);
-                               row.put(HTTP_METHOD, httpMethodValue);
-                               result.add(row);
-                       }
-              
-                       return result;
-               },
-               preparedStatement -> {
-                       preparedStatement.setString(1, userName);
-               });
-       }
-}
-
index 9b101fb..57d3516 100644 (file)
@@ -3,23 +3,30 @@ package com.prueba.authorization.services.impl;
 import java.util.List;
 import java.util.Map;
 
-import com.prueba.authorization.persistence.dao.AuthorizationDao;
+import com.prueba.authorization.persistence.dao.ApplicationResourceDao;
+import com.prueba.core.context.util.AntPathMatcher;
 
 public class AuthorizationServicesImpl {
-       private static final String USER_NAME_PARAM = "{username}";
+       private static final String USER_NAME_PARAM = "username";
+       private static final String API_URL_PATTERN = "/app/api/{" + USER_NAME_PARAM + "}";
 
-       public boolean isAuthorized(String httpMethod, String url, String userName) {
-               final AuthorizationDao dao = new AuthorizationDao();
+       public boolean isAuthorized(String httpMethod, String uri, String userName) {
+               final AntPathMatcher pathMatcher = new AntPathMatcher();
+               final Map<String, String> variables = pathMatcher.extractUriTemplateVariables(API_URL_PATTERN, uri);
+               final String userNameParam = variables.get(USER_NAME_PARAM);
+               
+               final ApplicationResourceDao dao = new ApplicationResourceDao();
                
                final List<Map<String, String>> urls = dao.findURLsByUserName(userName);
                
                return urls.stream().anyMatch(urlMap ->
                {
-                       final String urlPatternValue = urlMap.get(AuthorizationDao.URL_PATTERN);
-                       final String urlReplacedPatternValue = urlPatternValue.replace(USER_NAME_PARAM, userName);
-                       final String httpMethodValue = urlMap.get(AuthorizationDao.HTTP_METHOD);
+                       final String urlPatternValue = urlMap.get(ApplicationResourceDao.URL_PATTERN);
+                       final String urlReplacedPatternValue = urlPatternValue.replace("{" + USER_NAME_PARAM + "}", userNameParam);
+                       
+                       final String httpMethodValue = urlMap.get(ApplicationResourceDao.HTTP_METHOD);
                        
-                       return urlReplacedPatternValue.equals(url) && httpMethodValue.equals(httpMethod);
+                       return urlReplacedPatternValue.equals(uri) && httpMethodValue.equals(httpMethod);
                });
                
        }
diff --git a/src/main/java/com/prueba/core/context/util/AntPathMatcher.java b/src/main/java/com/prueba/core/context/util/AntPathMatcher.java
new file mode 100644 (file)
index 0000000..fa31423
--- /dev/null
@@ -0,0 +1,904 @@
+/*
+ * Copyright 2002-2016 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 com.prueba.core.context.util;
+
+
+
+import java.util.Comparator;
+import java.util.LinkedHashMap;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+/**
+ * {@link PathMatcher} implementation for Ant-style path patterns.
+ *
+ * <p>Part of this mapping code has been kindly borrowed from <a href="http://ant.apache.org">Apache Ant</a>.
+ *
+ * <p>The mapping matches URLs using the following rules:<br>
+ * <ul>
+ * <li>{@code ?} matches one character</li>
+ * <li>{@code *} matches zero or more characters</li>
+ * <li>{@code **} matches zero or more <em>directories</em> in a path</li>
+ * </ul>
+ *
+ * <h3>Examples</h3>
+ * <ul>
+ * <li>{@code com/t?st.jsp} &mdash; matches {@code com/test.jsp} but also
+ * {@code com/tast.jsp} or {@code com/txst.jsp}</li>
+ * <li>{@code com/*.jsp} &mdash; matches all {@code .jsp} files in the
+ * {@code com} directory</li>
+ * <li><code>com/&#42;&#42;/test.jsp</code> &mdash; matches all {@code test.jsp}
+ * files underneath the {@code com} path</li>
+ * <li><code>org/springframework/&#42;&#42;/*.jsp</code> &mdash; matches all
+ * {@code .jsp} files underneath the {@code org/springframework} path</li>
+ * <li><code>org/&#42;&#42;/servlet/bla.jsp</code> &mdash; matches
+ * {@code org/springframework/servlet/bla.jsp} but also
+ * {@code org/springframework/testing/servlet/bla.jsp} and {@code org/servlet/bla.jsp}</li>
+ * </ul>
+ *
+ * <p><strong>Note:</strong> a pattern and a path must both be absolute or must
+ * both be relative in order for the two to match. Therefore it is recommended
+ * that users of this implementation to sanitize patterns in order to prefix
+ * them with "/" as it makes sense in the context in which they're used.
+ *
+ * @author Alef Arendsen
+ * @author Juergen Hoeller
+ * @author Rob Harrop
+ * @author Arjen Poutsma
+ * @author Rossen Stoyanchev
+ * @author Sam Brannen
+ * @since 16.07.2003
+ */
+public class AntPathMatcher {
+
+       /** Default path separator: "/" */
+       public static final String DEFAULT_PATH_SEPARATOR = "/";
+
+       private static final int CACHE_TURNOFF_THRESHOLD = 65536;
+
+       private static final Pattern VARIABLE_PATTERN = Pattern.compile("\\{[^/]+?\\}");
+
+       private static final char[] WILDCARD_CHARS = { '*', '?', '{' };
+
+
+       private String pathSeparator;
+
+       private PathSeparatorPatternCache pathSeparatorPatternCache;
+
+       private boolean caseSensitive = true;
+
+       private boolean trimTokens = false;
+
+       private volatile Boolean cachePatterns;
+
+       private final Map<String, String[]> tokenizedPatternCache = new ConcurrentHashMap<String, String[]>(256);
+
+       final Map<String, AntPathStringMatcher> stringMatcherCache = new ConcurrentHashMap<String, AntPathStringMatcher>(256);
+
+
+       /**
+        * Create a new instance with the {@link #DEFAULT_PATH_SEPARATOR}.
+        */
+       public AntPathMatcher() {
+               this.pathSeparator = DEFAULT_PATH_SEPARATOR;
+               this.pathSeparatorPatternCache = new PathSeparatorPatternCache(DEFAULT_PATH_SEPARATOR);
+       }
+
+       /**
+        * A convenient, alternative constructor to use with a custom path separator.
+        * @param pathSeparator the path separator to use, must not be {@code null}.
+        * @since 4.1
+        */
+       public AntPathMatcher(String pathSeparator) {
+               this.pathSeparator = pathSeparator;
+               this.pathSeparatorPatternCache = new PathSeparatorPatternCache(pathSeparator);
+       }
+
+
+       /**
+        * Set the path separator to use for pattern parsing.
+        * <p>Default is "/", as in Ant.
+        */
+       public void setPathSeparator(String pathSeparator) {
+               this.pathSeparator = (pathSeparator != null ? pathSeparator : DEFAULT_PATH_SEPARATOR);
+               this.pathSeparatorPatternCache = new PathSeparatorPatternCache(this.pathSeparator);
+       }
+
+       /**
+        * Specify whether to perform pattern matching in a case-sensitive fashion.
+        * <p>Default is {@code true}. Switch this to {@code false} for case-insensitive matching.
+        * @since 4.2
+        */
+       public void setCaseSensitive(boolean caseSensitive) {
+               this.caseSensitive = caseSensitive;
+       }
+
+       /**
+        * Specify whether to trim tokenized paths and patterns.
+        * <p>Default is {@code true}.
+        */
+       public void setTrimTokens(boolean trimTokens) {
+               this.trimTokens = trimTokens;
+       }
+
+       /**
+        * Specify whether to cache parsed pattern metadata for patterns passed
+        * into this matcher's {@link #match} method. A value of {@code true}
+        * activates an unlimited pattern cache; a value of {@code false} turns
+        * the pattern cache off completely.
+        * <p>Default is for the cache to be on, but with the variant to automatically
+        * turn it off when encountering too many patterns to cache at runtime
+        * (the threshold is 65536), assuming that arbitrary permutations of patterns
+        * are coming in, with little chance for encountering a recurring pattern.
+        * @since 4.0.1
+        * @see #getStringMatcher(String)
+        */
+       public void setCachePatterns(boolean cachePatterns) {
+               this.cachePatterns = cachePatterns;
+       }
+
+       private void deactivatePatternCache() {
+               this.cachePatterns = false;
+               this.tokenizedPatternCache.clear();
+               this.stringMatcherCache.clear();
+       }
+
+
+       public boolean isPattern(String path) {
+               return (path.indexOf('*') != -1 || path.indexOf('?') != -1);
+       }
+
+       public boolean match(String pattern, String path) {
+               return doMatch(pattern, path, true, null);
+       }
+
+       public boolean matchStart(String pattern, String path) {
+               return doMatch(pattern, path, false, null);
+       }
+
+       /**
+        * Actually match the given {@code path} against the given {@code pattern}.
+        * @param pattern the pattern to match against
+        * @param path the path String to test
+        * @param fullMatch whether a full pattern match is required (else a pattern match
+        * as far as the given base path goes is sufficient)
+        * @return {@code true} if the supplied {@code path} matched, {@code false} if it didn't
+        */
+       protected boolean doMatch(String pattern, String path, boolean fullMatch, Map<String, String> uriTemplateVariables) {
+               if (path.startsWith(this.pathSeparator) != pattern.startsWith(this.pathSeparator)) {
+                       return false;
+               }
+
+               String[] pattDirs = tokenizePattern(pattern);
+               if (fullMatch && this.caseSensitive && !isPotentialMatch(path, pattDirs)) {
+                       return false;
+               }
+
+               String[] pathDirs = tokenizePath(path);
+
+               int pattIdxStart = 0;
+               int pattIdxEnd = pattDirs.length - 1;
+               int pathIdxStart = 0;
+               int pathIdxEnd = pathDirs.length - 1;
+
+               // Match all elements up to the first **
+               while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+                       String pattDir = pattDirs[pattIdxStart];
+                       if ("**".equals(pattDir)) {
+                               break;
+                       }
+                       if (!matchStrings(pattDir, pathDirs[pathIdxStart], uriTemplateVariables)) {
+                               return false;
+                       }
+                       pattIdxStart++;
+                       pathIdxStart++;
+               }
+
+               if (pathIdxStart > pathIdxEnd) {
+                       // Path is exhausted, only match if rest of pattern is * or **'s
+                       if (pattIdxStart > pattIdxEnd) {
+                               return (pattern.endsWith(this.pathSeparator) ? path.endsWith(this.pathSeparator) :
+                                               !path.endsWith(this.pathSeparator));
+                       }
+                       if (!fullMatch) {
+                               return true;
+                       }
+                       if (pattIdxStart == pattIdxEnd && pattDirs[pattIdxStart].equals("*") && path.endsWith(this.pathSeparator)) {
+                               return true;
+                       }
+                       for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+                               if (!pattDirs[i].equals("**")) {
+                                       return false;
+                               }
+                       }
+                       return true;
+               }
+               else if (pattIdxStart > pattIdxEnd) {
+                       // String not exhausted, but pattern is. Failure.
+                       return false;
+               }
+               else if (!fullMatch && "**".equals(pattDirs[pattIdxStart])) {
+                       // Path start definitely matches due to "**" part in pattern.
+                       return true;
+               }
+
+               // up to last '**'
+               while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+                       String pattDir = pattDirs[pattIdxEnd];
+                       if (pattDir.equals("**")) {
+                               break;
+                       }
+                       if (!matchStrings(pattDir, pathDirs[pathIdxEnd], uriTemplateVariables)) {
+                               return false;
+                       }
+                       pattIdxEnd--;
+                       pathIdxEnd--;
+               }
+               if (pathIdxStart > pathIdxEnd) {
+                       // String is exhausted
+                       for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+                               if (!pattDirs[i].equals("**")) {
+                                       return false;
+                               }
+                       }
+                       return true;
+               }
+
+               while (pattIdxStart != pattIdxEnd && pathIdxStart <= pathIdxEnd) {
+                       int patIdxTmp = -1;
+                       for (int i = pattIdxStart + 1; i <= pattIdxEnd; i++) {
+                               if (pattDirs[i].equals("**")) {
+                                       patIdxTmp = i;
+                                       break;
+                               }
+                       }
+                       if (patIdxTmp == pattIdxStart + 1) {
+                               // '**/**' situation, so skip one
+                               pattIdxStart++;
+                               continue;
+                       }
+                       // Find the pattern between padIdxStart & padIdxTmp in str between
+                       // strIdxStart & strIdxEnd
+                       int patLength = (patIdxTmp - pattIdxStart - 1);
+                       int strLength = (pathIdxEnd - pathIdxStart + 1);
+                       int foundIdx = -1;
+
+                       strLoop:
+                       for (int i = 0; i <= strLength - patLength; i++) {
+                               for (int j = 0; j < patLength; j++) {
+                                       String subPat = pattDirs[pattIdxStart + j + 1];
+                                       String subStr = pathDirs[pathIdxStart + i + j];
+                                       if (!matchStrings(subPat, subStr, uriTemplateVariables)) {
+                                               continue strLoop;
+                                       }
+                               }
+                               foundIdx = pathIdxStart + i;
+                               break;
+                       }
+
+                       if (foundIdx == -1) {
+                               return false;
+                       }
+
+                       pattIdxStart = patIdxTmp;
+                       pathIdxStart = foundIdx + patLength;
+               }
+
+               for (int i = pattIdxStart; i <= pattIdxEnd; i++) {
+                       if (!pattDirs[i].equals("**")) {
+                               return false;
+                       }
+               }
+
+               return true;
+       }
+
+       private boolean isPotentialMatch(String path, String[] pattDirs) {
+               if (!this.trimTokens) {
+                       char[] pathChars = path.toCharArray();
+                       int pos = 0;
+                       for (String pattDir : pattDirs) {
+                               int skipped = skipSeparator(path, pos, this.pathSeparator);
+                               pos += skipped;
+                               skipped = skipSegment(pathChars, pos, pattDir);
+                               if (skipped < pattDir.length()) {
+                                       if (skipped > 0) {
+                                               return true;
+                                       }
+                                       return (pattDir.length() > 0) && isWildcardChar(pattDir.charAt(0));
+                               }
+                               pos += skipped;
+                       }
+               }
+               return true;
+       }
+
+       private int skipSegment(char[] chars, int pos, String prefix) {
+               int skipped = 0;
+               for (char c : prefix.toCharArray()) {
+                       if (isWildcardChar(c)) {
+                               return skipped;
+                       }
+                       else if (pos + skipped >= chars.length) {
+                               return 0;
+                       }
+                       else if (chars[pos + skipped] == c) {
+                               skipped++;
+                       }
+               }
+               return skipped;
+       }
+
+       private int skipSeparator(String path, int pos, String separator) {
+               int skipped = 0;
+               while (path.startsWith(separator, pos + skipped)) {
+                       skipped += separator.length();
+               }
+               return skipped;
+       }
+
+       private boolean isWildcardChar(char c) {
+               for (char candidate : WILDCARD_CHARS) {
+                       if (c == candidate) {
+                               return true;
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Tokenize the given path pattern into parts, based on this matcher's settings.
+        * <p>Performs caching based on {@link #setCachePatterns}, delegating to
+        * {@link #tokenizePath(String)} for the actual tokenization algorithm.
+        * @param pattern the pattern to tokenize
+        * @return the tokenized pattern parts
+        */
+       protected String[] tokenizePattern(String pattern) {
+               String[] tokenized = null;
+               Boolean cachePatterns = this.cachePatterns;
+               if (cachePatterns == null || cachePatterns.booleanValue()) {
+                       tokenized = this.tokenizedPatternCache.get(pattern);
+               }
+               if (tokenized == null) {
+                       tokenized = tokenizePath(pattern);
+                       if (cachePatterns == null && this.tokenizedPatternCache.size() >= CACHE_TURNOFF_THRESHOLD) {
+                               // Try to adapt to the runtime situation that we're encountering:
+                               // There are obviously too many different patterns coming in here...
+                               // So let's turn off the cache since the patterns are unlikely to be reoccurring.
+                               deactivatePatternCache();
+                               return tokenized;
+                       }
+                       if (cachePatterns == null || cachePatterns.booleanValue()) {
+                               this.tokenizedPatternCache.put(pattern, tokenized);
+                       }
+               }
+               return tokenized;
+       }
+
+       /**
+        * Tokenize the given path String into parts, based on this matcher's settings.
+        * @param path the path to tokenize
+        * @return the tokenized path parts
+        */
+       protected String[] tokenizePath(String path) {
+               return StringUtils.tokenizeToStringArray(path, this.pathSeparator, this.trimTokens, true);
+       }
+
+       /**
+        * Test whether or not a string matches against a pattern.
+        * @param pattern the pattern to match against (never {@code null})
+        * @param str the String which must be matched against the pattern (never {@code null})
+        * @return {@code true} if the string matches against the pattern, or {@code false} otherwise
+        */
+       private boolean matchStrings(String pattern, String str, Map<String, String> uriTemplateVariables) {
+               return getStringMatcher(pattern).matchStrings(str, uriTemplateVariables);
+       }
+
+       /**
+        * Build or retrieve an {@link AntPathStringMatcher} for the given pattern.
+        * <p>The default implementation checks this AntPathMatcher's internal cache
+        * (see {@link #setCachePatterns}), creating a new AntPathStringMatcher instance
+        * if no cached copy is found.
+        * <p>When encountering too many patterns to cache at runtime (the threshold is 65536),
+        * it turns the default cache off, assuming that arbitrary permutations of patterns
+        * are coming in, with little chance for encountering a recurring pattern.
+        * <p>This method may be overridden to implement a custom cache strategy.
+        * @param pattern the pattern to match against (never {@code null})
+        * @return a corresponding AntPathStringMatcher (never {@code null})
+        * @see #setCachePatterns
+        */
+       protected AntPathStringMatcher getStringMatcher(String pattern) {
+               AntPathStringMatcher matcher = null;
+               Boolean cachePatterns = this.cachePatterns;
+               if (cachePatterns == null || cachePatterns.booleanValue()) {
+                       matcher = this.stringMatcherCache.get(pattern);
+               }
+               if (matcher == null) {
+                       matcher = new AntPathStringMatcher(pattern, this.caseSensitive);
+                       if (cachePatterns == null && this.stringMatcherCache.size() >= CACHE_TURNOFF_THRESHOLD) {
+                               // Try to adapt to the runtime situation that we're encountering:
+                               // There are obviously too many different patterns coming in here...
+                               // So let's turn off the cache since the patterns are unlikely to be reoccurring.
+                               deactivatePatternCache();
+                               return matcher;
+                       }
+                       if (cachePatterns == null || cachePatterns.booleanValue()) {
+                               this.stringMatcherCache.put(pattern, matcher);
+                       }
+               }
+               return matcher;
+       }
+
+       /**
+        * Given a pattern and a full path, determine the pattern-mapped part. <p>For example: <ul>
+        * <li>'{@code /docs/cvs/commit.html}' and '{@code /docs/cvs/commit.html} -> ''</li>
+        * <li>'{@code /docs/*}' and '{@code /docs/cvs/commit} -> '{@code cvs/commit}'</li>
+        * <li>'{@code /docs/cvs/*.html}' and '{@code /docs/cvs/commit.html} -> '{@code commit.html}'</li>
+        * <li>'{@code /docs/**}' and '{@code /docs/cvs/commit} -> '{@code cvs/commit}'</li>
+        * <li>'{@code /docs/**\/*.html}' and '{@code /docs/cvs/commit.html} -> '{@code cvs/commit.html}'</li>
+        * <li>'{@code /*.html}' and '{@code /docs/cvs/commit.html} -> '{@code docs/cvs/commit.html}'</li>
+        * <li>'{@code *.html}' and '{@code /docs/cvs/commit.html} -> '{@code /docs/cvs/commit.html}'</li>
+        * <li>'{@code *}' and '{@code /docs/cvs/commit.html} -> '{@code /docs/cvs/commit.html}'</li> </ul>
+        * <p>Assumes that {@link #match} returns {@code true} for '{@code pattern}' and '{@code path}', but
+        * does <strong>not</strong> enforce this.
+        */
+       public String extractPathWithinPattern(String pattern, String path) {
+               String[] patternParts = StringUtils.tokenizeToStringArray(pattern, this.pathSeparator, this.trimTokens, true);
+               String[] pathParts = StringUtils.tokenizeToStringArray(path, this.pathSeparator, this.trimTokens, true);
+               StringBuilder builder = new StringBuilder();
+               boolean pathStarted = false;
+
+               for (int segment = 0; segment < patternParts.length; segment++) {
+                       String patternPart = patternParts[segment];
+                       if (patternPart.indexOf('*') > -1 || patternPart.indexOf('?') > -1) {
+                               for (; segment < pathParts.length; segment++) {
+                                       if (pathStarted || (segment == 0 && !pattern.startsWith(this.pathSeparator))) {
+                                               builder.append(this.pathSeparator);
+                                       }
+                                       builder.append(pathParts[segment]);
+                                       pathStarted = true;
+                               }
+                       }
+               }
+
+               return builder.toString();
+       }
+
+       public Map<String, String> extractUriTemplateVariables(String pattern, String path) {
+               Map<String, String> variables = new LinkedHashMap<String, String>();
+               boolean result = doMatch(pattern, path, true, variables);
+               if (!result) {
+                       throw new IllegalStateException("Pattern \"" + pattern + "\" is not a match for \"" + path + "\"");
+               }
+               return variables;
+       }
+
+       /**
+        * Combine two patterns into a new pattern.
+        * <p>This implementation simply concatenates the two patterns, unless
+        * the first pattern contains a file extension match (e.g., {@code *.html}).
+        * In that case, the second pattern will be merged into the first. Otherwise,
+        * an {@code IllegalArgumentException} will be thrown.
+        * <h3>Examples</h3>
+        * <table border="1">
+        * <tr><th>Pattern 1</th><th>Pattern 2</th><th>Result</th></tr>
+        * <tr><td>{@code null}</td><td>{@code null}</td><td>&nbsp;</td></tr>
+        * <tr><td>/hotels</td><td>{@code null}</td><td>/hotels</td></tr>
+        * <tr><td>{@code null}</td><td>/hotels</td><td>/hotels</td></tr>
+        * <tr><td>/hotels</td><td>/bookings</td><td>/hotels/bookings</td></tr>
+        * <tr><td>/hotels</td><td>bookings</td><td>/hotels/bookings</td></tr>
+        * <tr><td>/hotels/*</td><td>/bookings</td><td>/hotels/bookings</td></tr>
+        * <tr><td>/hotels/&#42;&#42;</td><td>/bookings</td><td>/hotels/&#42;&#42;/bookings</td></tr>
+        * <tr><td>/hotels</td><td>{hotel}</td><td>/hotels/{hotel}</td></tr>
+        * <tr><td>/hotels/*</td><td>{hotel}</td><td>/hotels/{hotel}</td></tr>
+        * <tr><td>/hotels/&#42;&#42;</td><td>{hotel}</td><td>/hotels/&#42;&#42;/{hotel}</td></tr>
+        * <tr><td>/*.html</td><td>/hotels.html</td><td>/hotels.html</td></tr>
+        * <tr><td>/*.html</td><td>/hotels</td><td>/hotels.html</td></tr>
+        * <tr><td>/*.html</td><td>/*.txt</td><td>{@code IllegalArgumentException}</td></tr>
+        * </table>
+        * @param pattern1 the first pattern
+        * @param pattern2 the second pattern
+        * @return the combination of the two patterns
+        * @throws IllegalArgumentException if the two patterns cannot be combined
+        */
+       public String combine(String pattern1, String pattern2) {
+               if (!StringUtils.hasText(pattern1) && !StringUtils.hasText(pattern2)) {
+                       return "";
+               }
+               if (!StringUtils.hasText(pattern1)) {
+                       return pattern2;
+               }
+               if (!StringUtils.hasText(pattern2)) {
+                       return pattern1;
+               }
+
+               boolean pattern1ContainsUriVar = (pattern1.indexOf('{') != -1);
+               if (!pattern1.equals(pattern2) && !pattern1ContainsUriVar && match(pattern1, pattern2)) {
+                       // /* + /hotel -> /hotel ; "/*.*" + "/*.html" -> /*.html
+                       // However /user + /user -> /usr/user ; /{foo} + /bar -> /{foo}/bar
+                       return pattern2;
+               }
+
+               // /hotels/* + /booking -> /hotels/booking
+               // /hotels/* + booking -> /hotels/booking
+               if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnWildCard())) {
+                       return concat(pattern1.substring(0, pattern1.length() - 2), pattern2);
+               }
+
+               // /hotels/** + /booking -> /hotels/**/booking
+               // /hotels/** + booking -> /hotels/**/booking
+               if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnDoubleWildCard())) {
+                       return concat(pattern1, pattern2);
+               }
+
+               int starDotPos1 = pattern1.indexOf("*.");
+               if (pattern1ContainsUriVar || starDotPos1 == -1 || this.pathSeparator.equals(".")) {
+                       // simply concatenate the two patterns
+                       return concat(pattern1, pattern2);
+               }
+
+               String ext1 = pattern1.substring(starDotPos1 + 1);
+               int dotPos2 = pattern2.indexOf('.');
+               String file2 = (dotPos2 == -1 ? pattern2 : pattern2.substring(0, dotPos2));
+               String ext2 = (dotPos2 == -1 ? "" : pattern2.substring(dotPos2));
+               boolean ext1All = (ext1.equals(".*") || ext1.equals(""));
+               boolean ext2All = (ext2.equals(".*") || ext2.equals(""));
+               if (!ext1All && !ext2All) {
+                       throw new IllegalArgumentException("Cannot combine patterns: " + pattern1 + " vs " + pattern2);
+               }
+               String ext = (ext1All ? ext2 : ext1);
+               return file2 + ext;
+       }
+
+       private String concat(String path1, String path2) {
+               boolean path1EndsWithSeparator = path1.endsWith(this.pathSeparator);
+               boolean path2StartsWithSeparator = path2.startsWith(this.pathSeparator);
+
+               if (path1EndsWithSeparator && path2StartsWithSeparator) {
+                       return path1 + path2.substring(1);
+               }
+               else if (path1EndsWithSeparator || path2StartsWithSeparator) {
+                       return path1 + path2;
+               }
+               else {
+                       return path1 + this.pathSeparator + path2;
+               }
+       }
+
+       /**
+        * Given a full path, returns a {@link Comparator} suitable for sorting patterns in order of
+        * explicitness.
+        * <p>This{@code Comparator} will {@linkplain java.util.Collections#sort(List, Comparator) sort}
+        * a list so that more specific patterns (without uri templates or wild cards) come before
+        * generic patterns. So given a list with the following patterns:
+        * <ol>
+        * <li>{@code /hotels/new}</li>
+        * <li>{@code /hotels/{hotel}}</li> <li>{@code /hotels/*}</li>
+        * </ol>
+        * the returned comparator will sort this list so that the order will be as indicated.
+        * <p>The full path given as parameter is used to test for exact matches. So when the given path
+        * is {@code /hotels/2}, the pattern {@code /hotels/2} will be sorted before {@code /hotels/1}.
+        * @param path the full path to use for comparison
+        * @return a comparator capable of sorting patterns in order of explicitness
+        */
+       public Comparator<String> getPatternComparator(String path) {
+               return new AntPatternComparator(path);
+       }
+
+
+       /**
+        * Tests whether or not a string matches against a pattern via a {@link Pattern}.
+        * <p>The pattern may contain special characters: '*' means zero or more characters; '?' means one and
+        * only one character; '{' and '}' indicate a URI template pattern. For example <tt>/users/{user}</tt>.
+        */
+       protected static class AntPathStringMatcher {
+
+               private static final Pattern GLOB_PATTERN = Pattern.compile("\\?|\\*|\\{((?:\\{[^/]+?\\}|[^/{}]|\\\\[{}])+?)\\}");
+
+               private static final String DEFAULT_VARIABLE_PATTERN = "(.*)";
+
+               private final Pattern pattern;
+
+               private final List<String> variableNames = new LinkedList<String>();
+
+               public AntPathStringMatcher(String pattern) {
+                       this(pattern, true);
+               }
+
+               public AntPathStringMatcher(String pattern, boolean caseSensitive) {
+                       StringBuilder patternBuilder = new StringBuilder();
+                       Matcher matcher = GLOB_PATTERN.matcher(pattern);
+                       int end = 0;
+                       while (matcher.find()) {
+                               patternBuilder.append(quote(pattern, end, matcher.start()));
+                               String match = matcher.group();
+                               if ("?".equals(match)) {
+                                       patternBuilder.append('.');
+                               }
+                               else if ("*".equals(match)) {
+                                       patternBuilder.append(".*");
+                               }
+                               else if (match.startsWith("{") && match.endsWith("}")) {
+                                       int colonIdx = match.indexOf(':');
+                                       if (colonIdx == -1) {
+                                               patternBuilder.append(DEFAULT_VARIABLE_PATTERN);
+                                               this.variableNames.add(matcher.group(1));
+                                       }
+                                       else {
+                                               String variablePattern = match.substring(colonIdx + 1, match.length() - 1);
+                                               patternBuilder.append('(');
+                                               patternBuilder.append(variablePattern);
+                                               patternBuilder.append(')');
+                                               String variableName = match.substring(1, colonIdx);
+                                               this.variableNames.add(variableName);
+                                       }
+                               }
+                               end = matcher.end();
+                       }
+                       patternBuilder.append(quote(pattern, end, pattern.length()));
+                       this.pattern = (caseSensitive ? Pattern.compile(patternBuilder.toString()) :
+                                       Pattern.compile(patternBuilder.toString(), Pattern.CASE_INSENSITIVE));
+               }
+
+               private String quote(String s, int start, int end) {
+                       if (start == end) {
+                               return "";
+                       }
+                       return Pattern.quote(s.substring(start, end));
+               }
+
+               /**
+                * Main entry point.
+                * @return {@code true} if the string matches against the pattern, or {@code false} otherwise.
+                */
+               public boolean matchStrings(String str, Map<String, String> uriTemplateVariables) {
+                       Matcher matcher = this.pattern.matcher(str);
+                       if (matcher.matches()) {
+                               if (uriTemplateVariables != null) {
+                                       // SPR-8455
+                                       if (this.variableNames.size() != matcher.groupCount()) {
+                                               throw new IllegalArgumentException("The number of capturing groups in the pattern segment " +
+                                                               this.pattern + " does not match the number of URI template variables it defines, " +
+                                                               "which can occur if capturing groups are used in a URI template regex. " +
+                                                               "Use non-capturing groups instead.");
+                                       }
+                                       for (int i = 1; i <= matcher.groupCount(); i++) {
+                                               String name = this.variableNames.get(i - 1);
+                                               String value = matcher.group(i);
+                                               uriTemplateVariables.put(name, value);
+                                       }
+                               }
+                               return true;
+                       }
+                       else {
+                               return false;
+                       }
+               }
+       }
+
+
+       /**
+        * The default {@link Comparator} implementation returned by
+        * {@link #getPatternComparator(String)}.
+        * <p>In order, the most "generic" pattern is determined by the following:
+        * <ul>
+        * <li>if it's null or a capture all pattern (i.e. it is equal to "/**")</li>
+        * <li>if the other pattern is an actual match</li>
+        * <li>if it's a catch-all pattern (i.e. it ends with "**"</li>
+        * <li>if it's got more "*" than the other pattern</li>
+        * <li>if it's got more "{foo}" than the other pattern</li>
+        * <li>if it's shorter than the other pattern</li>
+        * </ul>
+        */
+       protected static class AntPatternComparator implements Comparator<String> {
+
+               private final String path;
+
+               public AntPatternComparator(String path) {
+                       this.path = path;
+               }
+
+               /**
+                * Compare two patterns to determine which should match first, i.e. which
+                * is the most specific regarding the current path.
+                * @return a negative integer, zero, or a positive integer as pattern1 is
+                * more specific, equally specific, or less specific than pattern2.
+                */
+               @Override
+               public int compare(String pattern1, String pattern2) {
+                       PatternInfo info1 = new PatternInfo(pattern1);
+                       PatternInfo info2 = new PatternInfo(pattern2);
+
+                       if (info1.isLeastSpecific() && info2.isLeastSpecific()) {
+                               return 0;
+                       }
+                       else if (info1.isLeastSpecific()) {
+                               return 1;
+                       }
+                       else if (info2.isLeastSpecific()) {
+                               return -1;
+                       }
+
+                       boolean pattern1EqualsPath = pattern1.equals(path);
+                       boolean pattern2EqualsPath = pattern2.equals(path);
+                       if (pattern1EqualsPath && pattern2EqualsPath) {
+                               return 0;
+                       }
+                       else if (pattern1EqualsPath) {
+                               return -1;
+                       }
+                       else if (pattern2EqualsPath) {
+                               return 1;
+                       }
+
+                       if (info1.isPrefixPattern() && info2.getDoubleWildcards() == 0) {
+                               return 1;
+                       }
+                       else if (info2.isPrefixPattern() && info1.getDoubleWildcards() == 0) {
+                               return -1;
+                       }
+
+                       if (info1.getTotalCount() != info2.getTotalCount()) {
+                               return info1.getTotalCount() - info2.getTotalCount();
+                       }
+
+                       if (info1.getLength() != info2.getLength()) {
+                               return info2.getLength() - info1.getLength();
+                       }
+
+                       if (info1.getSingleWildcards() < info2.getSingleWildcards()) {
+                               return -1;
+                       }
+                       else if (info2.getSingleWildcards() < info1.getSingleWildcards()) {
+                               return 1;
+                       }
+
+                       if (info1.getUriVars() < info2.getUriVars()) {
+                               return -1;
+                       }
+                       else if (info2.getUriVars() < info1.getUriVars()) {
+                               return 1;
+                       }
+
+                       return 0;
+               }
+
+
+               /**
+                * Value class that holds information about the pattern, e.g. number of
+                * occurrences of "*", "**", and "{" pattern elements.
+                */
+               private static class PatternInfo {
+
+                       private final String pattern;
+
+                       private int uriVars;
+
+                       private int singleWildcards;
+
+                       private int doubleWildcards;
+
+                       private boolean catchAllPattern;
+
+                       private boolean prefixPattern;
+
+                       private Integer length;
+
+                       public PatternInfo(String pattern) {
+                               this.pattern = pattern;
+                               if (this.pattern != null) {
+                                       initCounters();
+                                       this.catchAllPattern = this.pattern.equals("/**");
+                                       this.prefixPattern = !this.catchAllPattern && this.pattern.endsWith("/**");
+                               }
+                               if (this.uriVars == 0) {
+                                       this.length = (this.pattern != null ? this.pattern.length() : 0);
+                               }
+                       }
+
+                       protected void initCounters() {
+                               int pos = 0;
+                               while (pos < this.pattern.length()) {
+                                       if (this.pattern.charAt(pos) == '{') {
+                                               this.uriVars++;
+                                               pos++;
+                                       }
+                                       else if (this.pattern.charAt(pos) == '*') {
+                                               if (pos + 1 < this.pattern.length() && this.pattern.charAt(pos + 1) == '*') {
+                                                       this.doubleWildcards++;
+                                                       pos += 2;
+                                               }
+                                               else if (pos > 0 && !this.pattern.substring(pos - 1).equals(".*")) {
+                                                       this.singleWildcards++;
+                                                       pos++;
+                                               }
+                                               else {
+                                                       pos++;
+                                               }
+                                       }
+                                       else {
+                                               pos++;
+                                       }
+                               }
+                       }
+
+                       public int getUriVars() {
+                               return this.uriVars;
+                       }
+
+                       public int getSingleWildcards() {
+                               return this.singleWildcards;
+                       }
+
+                       public int getDoubleWildcards() {
+                               return this.doubleWildcards;
+                       }
+
+                       public boolean isLeastSpecific() {
+                               return (this.pattern == null || this.catchAllPattern);
+                       }
+
+                       public boolean isPrefixPattern() {
+                               return this.prefixPattern;
+                       }
+
+                       public int getTotalCount() {
+                               return this.uriVars + this.singleWildcards + (2 * this.doubleWildcards);
+                       }
+
+                       /**
+                        * Returns the length of the given pattern, where template variables are considered to be 1 long.
+                        */
+                       public int getLength() {
+                               if (this.length == null) {
+                                       this.length = VARIABLE_PATTERN.matcher(this.pattern).replaceAll("#").length();
+                               }
+                               return this.length;
+                       }
+               }
+       }
+
+
+       /**
+        * A simple cache for patterns that depend on the configured path separator.
+        */
+       private static class PathSeparatorPatternCache {
+
+               private final String endsOnWildCard;
+
+               private final String endsOnDoubleWildCard;
+
+               public PathSeparatorPatternCache(String pathSeparator) {
+                       this.endsOnWildCard = pathSeparator + "*";
+                       this.endsOnDoubleWildCard = pathSeparator + "**";
+               }
+
+               public String getEndsOnWildCard() {
+                       return this.endsOnWildCard;
+               }
+
+               public String getEndsOnDoubleWildCard() {
+                       return this.endsOnDoubleWildCard;
+               }
+       }
+
+}
+
diff --git a/src/main/java/com/prueba/core/context/util/CollectionUtils.java b/src/main/java/com/prueba/core/context/util/CollectionUtils.java
new file mode 100644 (file)
index 0000000..d6280bd
--- /dev/null
@@ -0,0 +1,26 @@
+package com.prueba.core.context.util;
+
+import java.util.Collection;
+
+/**
+ * Miscellaneous collection utility methods.
+ * Mainly for internal use within the framework.
+ *
+ * @author Juergen Hoeller
+ * @author Rob Harrop
+ * @author Arjen Poutsma
+ * @since 1.1.3
+ */
+public abstract class CollectionUtils {
+
+       /**
+        * Return {@code true} if the supplied Collection is {@code null} or empty.
+        * Otherwise, return {@code false}.
+        * @param collection the Collection to check
+        * @return whether the given Collection is empty
+        */
+       public static boolean isEmpty(Collection<?> collection) {
+               return (collection == null || collection.isEmpty());
+       }
+}
+
diff --git a/src/main/java/com/prueba/core/context/util/ObjectUtils.java b/src/main/java/com/prueba/core/context/util/ObjectUtils.java
new file mode 100644 (file)
index 0000000..0c6c59d
--- /dev/null
@@ -0,0 +1,881 @@
+package com.prueba.core.context.util;
+
+/*
+ * Copyright 2002-2013 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.
+ */
+
+import java.lang.reflect.Array;
+import java.util.Arrays;
+
+/**
+ * Miscellaneous object utility methods.
+ * Mainly for internal use within the framework.
+ *
+ * <p>Thanks to Alex Ruiz for contributing several enhancements to this class!
+ *
+ * @author Juergen Hoeller
+ * @author Keith Donald
+ * @author Rod Johnson
+ * @author Rob Harrop
+ * @author Chris Beams
+ * @since 19.03.2004
+ */
+public abstract class ObjectUtils {
+
+       private static final int INITIAL_HASH = 7;
+       private static final int MULTIPLIER = 31;
+
+       private static final String EMPTY_STRING = "";
+       private static final String NULL_STRING = "null";
+       private static final String ARRAY_START = "{";
+       private static final String ARRAY_END = "}";
+       private static final String EMPTY_ARRAY = ARRAY_START + ARRAY_END;
+       private static final String ARRAY_ELEMENT_SEPARATOR = ", ";
+
+
+       /**
+        * Return whether the given throwable is a checked exception:
+        * that is, neither a RuntimeException nor an Error.
+        * @param ex the throwable to check
+        * @return whether the throwable is a checked exception
+        * @see java.lang.Exception
+        * @see java.lang.RuntimeException
+        * @see java.lang.Error
+        */
+       public static boolean isCheckedException(Throwable ex) {
+               return !(ex instanceof RuntimeException || ex instanceof Error);
+       }
+
+       /**
+        * Check whether the given exception is compatible with the specified
+        * exception types, as declared in a throws clause.
+        * @param ex the exception to check
+        * @param declaredExceptions the exception types declared in the throws clause
+        * @return whether the given exception is compatible
+        */
+       public static boolean isCompatibleWithThrowsClause(Throwable ex, Class<?>... declaredExceptions) {
+               if (!isCheckedException(ex)) {
+                       return true;
+               }
+               if (declaredExceptions != null) {
+                       for (Class<?> declaredException : declaredExceptions) {
+                               if (declaredException.isInstance(ex)) {
+                                       return true;
+                               }
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Determine whether the given object is an array:
+        * either an Object array or a primitive array.
+        * @param obj the object to check
+        */
+       public static boolean isArray(Object obj) {
+               return (obj != null && obj.getClass().isArray());
+       }
+
+       /**
+        * Determine whether the given array is empty:
+        * i.e. {@code null} or of zero length.
+        * @param array the array to check
+        */
+       public static boolean isEmpty(Object[] array) {
+               return (array == null || array.length == 0);
+       }
+
+       /**
+        * Check whether the given array contains the given element.
+        * @param array the array to check (may be {@code null},
+        * in which case the return value will always be {@code false})
+        * @param element the element to check for
+        * @return whether the element has been found in the given array
+        */
+       public static boolean containsElement(Object[] array, Object element) {
+               if (array == null) {
+                       return false;
+               }
+               for (Object arrayEle : array) {
+                       if (nullSafeEquals(arrayEle, element)) {
+                               return true;
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Check whether the given array of enum constants contains a constant with the given name,
+        * ignoring case when determining a match.
+        * @param enumValues the enum values to check, typically the product of a call to MyEnum.values()
+        * @param constant the constant name to find (must not be null or empty string)
+        * @return whether the constant has been found in the given array
+        */
+       public static boolean containsConstant(Enum<?>[] enumValues, String constant) {
+               return containsConstant(enumValues, constant, false);
+       }
+
+       /**
+        * Check whether the given array of enum constants contains a constant with the given name.
+        * @param enumValues the enum values to check, typically the product of a call to MyEnum.values()
+        * @param constant the constant name to find (must not be null or empty string)
+        * @param caseSensitive whether case is significant in determining a match
+        * @return whether the constant has been found in the given array
+        */
+       public static boolean containsConstant(Enum<?>[] enumValues, String constant, boolean caseSensitive) {
+               for (Enum<?> candidate : enumValues) {
+                       if (caseSensitive ?
+                                       candidate.toString().equals(constant) :
+                                       candidate.toString().equalsIgnoreCase(constant)) {
+                               return true;
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Case insensitive alternative to {@link Enum#valueOf(Class, String)}.
+        * @param <E> the concrete Enum type
+        * @param enumValues the array of all Enum constants in question, usually per Enum.values()
+        * @param constant the constant to get the enum value of
+        * @throws IllegalArgumentException if the given constant is not found in the given array
+        * of enum values. Use {@link #containsConstant(Enum[], String)} as a guard to avoid this exception.
+        */
+       public static <E extends Enum<?>> E caseInsensitiveValueOf(E[] enumValues, String constant) {
+               for (E candidate : enumValues) {
+                       if (candidate.toString().equalsIgnoreCase(constant)) {
+                               return candidate;
+                       }
+               }
+               throw new IllegalArgumentException(
+                               String.format("constant [%s] does not exist in enum type %s",
+                                               constant, enumValues.getClass().getComponentType().getName()));
+       }
+
+       /**
+        * Append the given object to the given array, returning a new array
+        * consisting of the input array contents plus the given object.
+        * @param array the array to append to (can be {@code null})
+        * @param obj the object to append
+        * @return the new array (of the same component type; never {@code null})
+        */
+       public static <A, O extends A> A[] addObjectToArray(A[] array, O obj) {
+               Class<?> compType = Object.class;
+               if (array != null) {
+                       compType = array.getClass().getComponentType();
+               }
+               else if (obj != null) {
+                       compType = obj.getClass();
+               }
+               int newArrLength = (array != null ? array.length + 1 : 1);
+               @SuppressWarnings("unchecked")
+               A[] newArr = (A[]) Array.newInstance(compType, newArrLength);
+               if (array != null) {
+                       System.arraycopy(array, 0, newArr, 0, array.length);
+               }
+               newArr[newArr.length - 1] = obj;
+               return newArr;
+       }
+
+       /**
+        * Convert the given array (which may be a primitive array) to an
+        * object array (if necessary of primitive wrapper objects).
+        * <p>A {@code null} source value will be converted to an
+        * empty Object array.
+        * @param source the (potentially primitive) array
+        * @return the corresponding object array (never {@code null})
+        * @throws IllegalArgumentException if the parameter is not an array
+        */
+       public static Object[] toObjectArray(Object source) {
+               if (source instanceof Object[]) {
+                       return (Object[]) source;
+               }
+               if (source == null) {
+                       return new Object[0];
+               }
+               if (!source.getClass().isArray()) {
+                       throw new IllegalArgumentException("Source is not an array: " + source);
+               }
+               int length = Array.getLength(source);
+               if (length == 0) {
+                       return new Object[0];
+               }
+               Class<?> wrapperType = Array.get(source, 0).getClass();
+               Object[] newArray = (Object[]) Array.newInstance(wrapperType, length);
+               for (int i = 0; i < length; i++) {
+                       newArray[i] = Array.get(source, i);
+               }
+               return newArray;
+       }
+
+
+       //---------------------------------------------------------------------
+       // Convenience methods for content-based equality/hash-code handling
+       //---------------------------------------------------------------------
+
+       /**
+        * Determine if the given objects are equal, returning {@code true}
+        * if both are {@code null} or {@code false} if only one is
+        * {@code null}.
+        * <p>Compares arrays with {@code Arrays.equals}, performing an equality
+        * check based on the array elements rather than the array reference.
+        * @param o1 first Object to compare
+        * @param o2 second Object to compare
+        * @return whether the given objects are equal
+        * @see java.util.Arrays#equals
+        */
+       public static boolean nullSafeEquals(Object o1, Object o2) {
+               if (o1 == o2) {
+                       return true;
+               }
+               if (o1 == null || o2 == null) {
+                       return false;
+               }
+               if (o1.equals(o2)) {
+                       return true;
+               }
+               if (o1.getClass().isArray() && o2.getClass().isArray()) {
+                       if (o1 instanceof Object[] && o2 instanceof Object[]) {
+                               return Arrays.equals((Object[]) o1, (Object[]) o2);
+                       }
+                       if (o1 instanceof boolean[] && o2 instanceof boolean[]) {
+                               return Arrays.equals((boolean[]) o1, (boolean[]) o2);
+                       }
+                       if (o1 instanceof byte[] && o2 instanceof byte[]) {
+                               return Arrays.equals((byte[]) o1, (byte[]) o2);
+                       }
+                       if (o1 instanceof char[] && o2 instanceof char[]) {
+                               return Arrays.equals((char[]) o1, (char[]) o2);
+                       }
+                       if (o1 instanceof double[] && o2 instanceof double[]) {
+                               return Arrays.equals((double[]) o1, (double[]) o2);
+                       }
+                       if (o1 instanceof float[] && o2 instanceof float[]) {
+                               return Arrays.equals((float[]) o1, (float[]) o2);
+                       }
+                       if (o1 instanceof int[] && o2 instanceof int[]) {
+                               return Arrays.equals((int[]) o1, (int[]) o2);
+                       }
+                       if (o1 instanceof long[] && o2 instanceof long[]) {
+                               return Arrays.equals((long[]) o1, (long[]) o2);
+                       }
+                       if (o1 instanceof short[] && o2 instanceof short[]) {
+                               return Arrays.equals((short[]) o1, (short[]) o2);
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Return as hash code for the given object; typically the value of
+        * {@code Object#hashCode()}}. If the object is an array,
+        * this method will delegate to any of the {@code nullSafeHashCode}
+        * methods for arrays in this class. If the object is {@code null},
+        * this method returns 0.
+        * @see #nullSafeHashCode(Object[])
+        * @see #nullSafeHashCode(boolean[])
+        * @see #nullSafeHashCode(byte[])
+        * @see #nullSafeHashCode(char[])
+        * @see #nullSafeHashCode(double[])
+        * @see #nullSafeHashCode(float[])
+        * @see #nullSafeHashCode(int[])
+        * @see #nullSafeHashCode(long[])
+        * @see #nullSafeHashCode(short[])
+        */
+       public static int nullSafeHashCode(Object obj) {
+               if (obj == null) {
+                       return 0;
+               }
+               if (obj.getClass().isArray()) {
+                       if (obj instanceof Object[]) {
+                               return nullSafeHashCode((Object[]) obj);
+                       }
+                       if (obj instanceof boolean[]) {
+                               return nullSafeHashCode((boolean[]) obj);
+                       }
+                       if (obj instanceof byte[]) {
+                               return nullSafeHashCode((byte[]) obj);
+                       }
+                       if (obj instanceof char[]) {
+                               return nullSafeHashCode((char[]) obj);
+                       }
+                       if (obj instanceof double[]) {
+                               return nullSafeHashCode((double[]) obj);
+                       }
+                       if (obj instanceof float[]) {
+                               return nullSafeHashCode((float[]) obj);
+                       }
+                       if (obj instanceof int[]) {
+                               return nullSafeHashCode((int[]) obj);
+                       }
+                       if (obj instanceof long[]) {
+                               return nullSafeHashCode((long[]) obj);
+                       }
+                       if (obj instanceof short[]) {
+                               return nullSafeHashCode((short[]) obj);
+                       }
+               }
+               return obj.hashCode();
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(Object[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (Object element : array) {
+                       hash = MULTIPLIER * hash + nullSafeHashCode(element);
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(boolean[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (boolean element : array) {
+                       hash = MULTIPLIER * hash + hashCode(element);
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(byte[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (byte element : array) {
+                       hash = MULTIPLIER * hash + element;
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(char[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (char element : array) {
+                       hash = MULTIPLIER * hash + element;
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(double[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (double element : array) {
+                       hash = MULTIPLIER * hash + hashCode(element);
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(float[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (float element : array) {
+                       hash = MULTIPLIER * hash + hashCode(element);
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(int[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (int element : array) {
+                       hash = MULTIPLIER * hash + element;
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(long[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (long element : array) {
+                       hash = MULTIPLIER * hash + hashCode(element);
+               }
+               return hash;
+       }
+
+       /**
+        * Return a hash code based on the contents of the specified array.
+        * If {@code array} is {@code null}, this method returns 0.
+        */
+       public static int nullSafeHashCode(short[] array) {
+               if (array == null) {
+                       return 0;
+               }
+               int hash = INITIAL_HASH;
+               for (short element : array) {
+                       hash = MULTIPLIER * hash + element;
+               }
+               return hash;
+       }
+
+       /**
+        * Return the same value as {@link Boolean#hashCode()}}.
+        * @see Boolean#hashCode()
+        */
+       public static int hashCode(boolean bool) {
+               return (bool ? 1231 : 1237);
+       }
+
+       /**
+        * Return the same value as {@link Double#hashCode()}}.
+        * @see Double#hashCode()
+        */
+       public static int hashCode(double dbl) {
+               return hashCode(Double.doubleToLongBits(dbl));
+       }
+
+       /**
+        * Return the same value as {@link Float#hashCode()}}.
+        * @see Float#hashCode()
+        */
+       public static int hashCode(float flt) {
+               return Float.floatToIntBits(flt);
+       }
+
+       /**
+        * Return the same value as {@link Long#hashCode()}}.
+        * @see Long#hashCode()
+        */
+       public static int hashCode(long lng) {
+               return (int) (lng ^ (lng >>> 32));
+       }
+
+
+       //---------------------------------------------------------------------
+       // Convenience methods for toString output
+       //---------------------------------------------------------------------
+
+       /**
+        * Return a String representation of an object's overall identity.
+        * @param obj the object (may be {@code null})
+        * @return the object's identity as String representation,
+        * or an empty String if the object was {@code null}
+        */
+       public static String identityToString(Object obj) {
+               if (obj == null) {
+                       return EMPTY_STRING;
+               }
+               return obj.getClass().getName() + "@" + getIdentityHexString(obj);
+       }
+
+       /**
+        * Return a hex String form of an object's identity hash code.
+        * @param obj the object
+        * @return the object's identity code in hex notation
+        */
+       public static String getIdentityHexString(Object obj) {
+               return Integer.toHexString(System.identityHashCode(obj));
+       }
+
+       /**
+        * Return a content-based String representation if {@code obj} is
+        * not {@code null}; otherwise returns an empty String.
+        * <p>Differs from {@link #nullSafeToString(Object)} in that it returns
+        * an empty String rather than "null" for a {@code null} value.
+        * @param obj the object to build a display String for
+        * @return a display String representation of {@code obj}
+        * @see #nullSafeToString(Object)
+        */
+       public static String getDisplayString(Object obj) {
+               if (obj == null) {
+                       return EMPTY_STRING;
+               }
+               return nullSafeToString(obj);
+       }
+
+       /**
+        * Determine the class name for the given object.
+        * <p>Returns {@code "null"} if {@code obj} is {@code null}.
+        * @param obj the object to introspect (may be {@code null})
+        * @return the corresponding class name
+        */
+       public static String nullSafeClassName(Object obj) {
+               return (obj != null ? obj.getClass().getName() : NULL_STRING);
+       }
+
+       /**
+        * Return a String representation of the specified Object.
+        * <p>Builds a String representation of the contents in case of an array.
+        * Returns {@code "null"} if {@code obj} is {@code null}.
+        * @param obj the object to build a String representation for
+        * @return a String representation of {@code obj}
+        */
+       public static String nullSafeToString(Object obj) {
+               if (obj == null) {
+                       return NULL_STRING;
+               }
+               if (obj instanceof String) {
+                       return (String) obj;
+               }
+               if (obj instanceof Object[]) {
+                       return nullSafeToString((Object[]) obj);
+               }
+               if (obj instanceof boolean[]) {
+                       return nullSafeToString((boolean[]) obj);
+               }
+               if (obj instanceof byte[]) {
+                       return nullSafeToString((byte[]) obj);
+               }
+               if (obj instanceof char[]) {
+                       return nullSafeToString((char[]) obj);
+               }
+               if (obj instanceof double[]) {
+                       return nullSafeToString((double[]) obj);
+               }
+               if (obj instanceof float[]) {
+                       return nullSafeToString((float[]) obj);
+               }
+               if (obj instanceof int[]) {
+                       return nullSafeToString((int[]) obj);
+               }
+               if (obj instanceof long[]) {
+                       return nullSafeToString((long[]) obj);
+               }
+               if (obj instanceof short[]) {
+                       return nullSafeToString((short[]) obj);
+               }
+               String str = obj.toString();
+               return (str != null ? str : EMPTY_STRING);
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(Object[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append(String.valueOf(array[i]));
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(boolean[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(byte[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(char[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append("'").append(array[i]).append("'");
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(double[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(float[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(int[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(long[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+       /**
+        * Return a String representation of the contents of the specified array.
+        * <p>The String representation consists of a list of the array's elements,
+        * enclosed in curly braces ({@code "{}"}). Adjacent elements are separated
+        * by the characters {@code ", "} (a comma followed by a space). Returns
+        * {@code "null"} if {@code array} is {@code null}.
+        * @param array the array to build a String representation for
+        * @return a String representation of {@code array}
+        */
+       public static String nullSafeToString(short[] array) {
+               if (array == null) {
+                       return NULL_STRING;
+               }
+               int length = array.length;
+               if (length == 0) {
+                       return EMPTY_ARRAY;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < length; i++) {
+                       if (i == 0) {
+                               sb.append(ARRAY_START);
+                       }
+                       else {
+                               sb.append(ARRAY_ELEMENT_SEPARATOR);
+                       }
+                       sb.append(array[i]);
+               }
+               sb.append(ARRAY_END);
+               return sb.toString();
+       }
+
+}
+
diff --git a/src/main/java/com/prueba/core/context/util/StringUtils.java b/src/main/java/com/prueba/core/context/util/StringUtils.java
new file mode 100644 (file)
index 0000000..72033f4
--- /dev/null
@@ -0,0 +1,1178 @@
+package com.prueba.core.context.util;
+
+/*
+ * Copyright 2002-2015 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.
+ */
+
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Enumeration;
+import java.util.Iterator;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Locale;
+import java.util.Properties;
+import java.util.Set;
+import java.util.StringTokenizer;
+import java.util.TimeZone;
+import java.util.TreeSet;
+
+/**
+ * Miscellaneous {@link String} utility methods.
+ *
+ * <p>Mainly for internal use within the framework; consider
+ * <a href="http://jakarta.apache.org/commons/lang/">Apache's Commons Lang</a>
+ * for a more comprehensive suite of String utilities.
+ *
+ * <p>This class delivers some simple functionality that should really
+ * be provided by the core Java {@code String} and {@link StringBuilder}
+ * classes, such as the ability to {@link #replace} all occurrences of a given
+ * substring in a target string. It also provides easy-to-use methods to convert
+ * between delimited strings, such as CSV strings, and collections and arrays.
+ *
+ * @author Rod Johnson
+ * @author Juergen Hoeller
+ * @author Keith Donald
+ * @author Rob Harrop
+ * @author Rick Evans
+ * @author Arjen Poutsma
+ * @since 16 April 2001
+ */
+public abstract class StringUtils {
+
+       private static final String FOLDER_SEPARATOR = "/";
+
+       private static final String WINDOWS_FOLDER_SEPARATOR = "\\";
+
+       private static final String TOP_PATH = "..";
+
+       private static final String CURRENT_PATH = ".";
+
+       private static final char EXTENSION_SEPARATOR = '.';
+
+
+       //---------------------------------------------------------------------
+       // General convenience methods for working with Strings
+       //---------------------------------------------------------------------
+
+       /**
+        * Check whether the given String is empty.
+        * <p>This method accepts any Object as an argument, comparing it to
+        * {@code null} and the empty String. As a consequence, this method
+        * will never return {@code true} for a non-null non-String object.
+        * <p>The Object signature is useful for general attribute handling code
+        * that commonly deals with Strings but generally has to iterate over
+        * Objects since attributes may e.g. be primitive value objects as well.
+        * @param str the candidate String
+        * @since 3.2.1
+        */
+       public static boolean isEmpty(Object str) {
+               return (str == null || "".equals(str));
+       }
+
+       /**
+        * Check that the given CharSequence is neither {@code null} nor of length 0.
+        * Note: Will return {@code true} for a CharSequence that purely consists of whitespace.
+        * <p><pre class="code">
+        * StringUtils.hasLength(null) = false
+        * StringUtils.hasLength("") = false
+        * StringUtils.hasLength(" ") = true
+        * StringUtils.hasLength("Hello") = true
+        * </pre>
+        * @param str the CharSequence to check (may be {@code null})
+        * @return {@code true} if the CharSequence is not null and has length
+        * @see #hasText(String)
+        */
+       public static boolean hasLength(CharSequence str) {
+               return (str != null && str.length() > 0);
+       }
+
+       /**
+        * Check that the given String is neither {@code null} nor of length 0.
+        * Note: Will return {@code true} for a String that purely consists of whitespace.
+        * @param str the String to check (may be {@code null})
+        * @return {@code true} if the String is not null and has length
+        * @see #hasLength(CharSequence)
+        */
+       public static boolean hasLength(String str) {
+               return hasLength((CharSequence) str);
+       }
+
+       /**
+        * Check whether the given CharSequence has actual text.
+        * More specifically, returns {@code true} if the string not {@code null},
+        * its length is greater than 0, and it contains at least one non-whitespace character.
+        * <p><pre class="code">
+        * StringUtils.hasText(null) = false
+        * StringUtils.hasText("") = false
+        * StringUtils.hasText(" ") = false
+        * StringUtils.hasText("12345") = true
+        * StringUtils.hasText(" 12345 ") = true
+        * </pre>
+        * @param str the CharSequence to check (may be {@code null})
+        * @return {@code true} if the CharSequence is not {@code null},
+        * its length is greater than 0, and it does not contain whitespace only
+        * @see Character#isWhitespace
+        */
+       public static boolean hasText(CharSequence str) {
+               if (!hasLength(str)) {
+                       return false;
+               }
+               int strLen = str.length();
+               for (int i = 0; i < strLen; i++) {
+                       if (!Character.isWhitespace(str.charAt(i))) {
+                               return true;
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Check whether the given String has actual text.
+        * More specifically, returns {@code true} if the string not {@code null},
+        * its length is greater than 0, and it contains at least one non-whitespace character.
+        * @param str the String to check (may be {@code null})
+        * @return {@code true} if the String is not {@code null}, its length is
+        * greater than 0, and it does not contain whitespace only
+        * @see #hasText(CharSequence)
+        */
+       public static boolean hasText(String str) {
+               return hasText((CharSequence) str);
+       }
+
+       /**
+        * Check whether the given CharSequence contains any whitespace characters.
+        * @param str the CharSequence to check (may be {@code null})
+        * @return {@code true} if the CharSequence is not empty and
+        * contains at least 1 whitespace character
+        * @see Character#isWhitespace
+        */
+       public static boolean containsWhitespace(CharSequence str) {
+               if (!hasLength(str)) {
+                       return false;
+               }
+               int strLen = str.length();
+               for (int i = 0; i < strLen; i++) {
+                       if (Character.isWhitespace(str.charAt(i))) {
+                               return true;
+                       }
+               }
+               return false;
+       }
+
+       /**
+        * Check whether the given String contains any whitespace characters.
+        * @param str the String to check (may be {@code null})
+        * @return {@code true} if the String is not empty and
+        * contains at least 1 whitespace character
+        * @see #containsWhitespace(CharSequence)
+        */
+       public static boolean containsWhitespace(String str) {
+               return containsWhitespace((CharSequence) str);
+       }
+
+       /**
+        * Trim leading and trailing whitespace from the given String.
+        * @param str the String to check
+        * @return the trimmed String
+        * @see java.lang.Character#isWhitespace
+        */
+       public static String trimWhitespace(String str) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str);
+               while (sb.length() > 0 && Character.isWhitespace(sb.charAt(0))) {
+                       sb.deleteCharAt(0);
+               }
+               while (sb.length() > 0 && Character.isWhitespace(sb.charAt(sb.length() - 1))) {
+                       sb.deleteCharAt(sb.length() - 1);
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Trim <i>all</i> whitespace from the given String:
+        * leading, trailing, and in between characters.
+        * @param str the String to check
+        * @return the trimmed String
+        * @see java.lang.Character#isWhitespace
+        */
+       public static String trimAllWhitespace(String str) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               int len = str.length();
+               StringBuilder sb = new StringBuilder(str.length());
+               for (int i = 0; i < len; i++) {
+                       char c = str.charAt(i);
+                       if (!Character.isWhitespace(c)) {
+                               sb.append(c);
+                       }
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Trim leading whitespace from the given String.
+        * @param str the String to check
+        * @return the trimmed String
+        * @see java.lang.Character#isWhitespace
+        */
+       public static String trimLeadingWhitespace(String str) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str);
+               while (sb.length() > 0 && Character.isWhitespace(sb.charAt(0))) {
+                       sb.deleteCharAt(0);
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Trim trailing whitespace from the given String.
+        * @param str the String to check
+        * @return the trimmed String
+        * @see java.lang.Character#isWhitespace
+        */
+       public static String trimTrailingWhitespace(String str) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str);
+               while (sb.length() > 0 && Character.isWhitespace(sb.charAt(sb.length() - 1))) {
+                       sb.deleteCharAt(sb.length() - 1);
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Trim all occurrences of the supplied leading character from the given String.
+        * @param str the String to check
+        * @param leadingCharacter the leading character to be trimmed
+        * @return the trimmed String
+        */
+       public static String trimLeadingCharacter(String str, char leadingCharacter) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str);
+               while (sb.length() > 0 && sb.charAt(0) == leadingCharacter) {
+                       sb.deleteCharAt(0);
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Trim all occurrences of the supplied trailing character from the given String.
+        * @param str the String to check
+        * @param trailingCharacter the trailing character to be trimmed
+        * @return the trimmed String
+        */
+       public static String trimTrailingCharacter(String str, char trailingCharacter) {
+               if (!hasLength(str)) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str);
+               while (sb.length() > 0 && sb.charAt(sb.length() - 1) == trailingCharacter) {
+                       sb.deleteCharAt(sb.length() - 1);
+               }
+               return sb.toString();
+       }
+
+
+       /**
+        * Test if the given String starts with the specified prefix,
+        * ignoring upper/lower case.
+        * @param str the String to check
+        * @param prefix the prefix to look for
+        * @see java.lang.String#startsWith
+        */
+       public static boolean startsWithIgnoreCase(String str, String prefix) {
+               if (str == null || prefix == null) {
+                       return false;
+               }
+               if (str.startsWith(prefix)) {
+                       return true;
+               }
+               if (str.length() < prefix.length()) {
+                       return false;
+               }
+               String lcStr = str.substring(0, prefix.length()).toLowerCase();
+               String lcPrefix = prefix.toLowerCase();
+               return lcStr.equals(lcPrefix);
+       }
+
+       /**
+        * Test if the given String ends with the specified suffix,
+        * ignoring upper/lower case.
+        * @param str the String to check
+        * @param suffix the suffix to look for
+        * @see java.lang.String#endsWith
+        */
+       public static boolean endsWithIgnoreCase(String str, String suffix) {
+               if (str == null || suffix == null) {
+                       return false;
+               }
+               if (str.endsWith(suffix)) {
+                       return true;
+               }
+               if (str.length() < suffix.length()) {
+                       return false;
+               }
+
+               String lcStr = str.substring(str.length() - suffix.length()).toLowerCase();
+               String lcSuffix = suffix.toLowerCase();
+               return lcStr.equals(lcSuffix);
+       }
+
+       /**
+        * Test whether the given string matches the given substring
+        * at the given index.
+        * @param str the original string (or StringBuilder)
+        * @param index the index in the original string to start matching against
+        * @param substring the substring to match at the given index
+        */
+       public static boolean substringMatch(CharSequence str, int index, CharSequence substring) {
+               for (int j = 0; j < substring.length(); j++) {
+                       int i = index + j;
+                       if (i >= str.length() || str.charAt(i) != substring.charAt(j)) {
+                               return false;
+                       }
+               }
+               return true;
+       }
+
+       /**
+        * Count the occurrences of the substring in string s.
+        * @param str string to search in. Return 0 if this is null.
+        * @param sub string to search for. Return 0 if this is null.
+        */
+       public static int countOccurrencesOf(String str, String sub) {
+               if (str == null || sub == null || str.length() == 0 || sub.length() == 0) {
+                       return 0;
+               }
+               int count = 0;
+               int pos = 0;
+               int idx;
+               while ((idx = str.indexOf(sub, pos)) != -1) {
+                       ++count;
+                       pos = idx + sub.length();
+               }
+               return count;
+       }
+
+       /**
+        * Replace all occurrences of a substring within a string with
+        * another string.
+        * @param inString String to examine
+        * @param oldPattern String to replace
+        * @param newPattern String to insert
+        * @return a String with the replacements
+        */
+       public static String replace(String inString, String oldPattern, String newPattern) {
+               if (!hasLength(inString) || !hasLength(oldPattern) || newPattern == null) {
+                       return inString;
+               }
+               StringBuilder sb = new StringBuilder();
+               int pos = 0; // our position in the old string
+               int index = inString.indexOf(oldPattern);
+               // the index of an occurrence we've found, or -1
+               int patLen = oldPattern.length();
+               while (index >= 0) {
+                       sb.append(inString.substring(pos, index));
+                       sb.append(newPattern);
+                       pos = index + patLen;
+                       index = inString.indexOf(oldPattern, pos);
+               }
+               sb.append(inString.substring(pos));
+               // remember to append any characters to the right of a match
+               return sb.toString();
+       }
+
+       /**
+        * Delete all occurrences of the given substring.
+        * @param inString the original String
+        * @param pattern the pattern to delete all occurrences of
+        * @return the resulting String
+        */
+       public static String delete(String inString, String pattern) {
+               return replace(inString, pattern, "");
+       }
+
+       /**
+        * Delete any character in a given String.
+        * @param inString the original String
+        * @param charsToDelete a set of characters to delete.
+        * E.g. "az\n" will delete 'a's, 'z's and new lines.
+        * @return the resulting String
+        */
+       public static String deleteAny(String inString, String charsToDelete) {
+               if (!hasLength(inString) || !hasLength(charsToDelete)) {
+                       return inString;
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < inString.length(); i++) {
+                       char c = inString.charAt(i);
+                       if (charsToDelete.indexOf(c) == -1) {
+                               sb.append(c);
+                       }
+               }
+               return sb.toString();
+       }
+
+
+       //---------------------------------------------------------------------
+       // Convenience methods for working with formatted Strings
+       //---------------------------------------------------------------------
+
+       /**
+        * Quote the given String with single quotes.
+        * @param str the input String (e.g. "myString")
+        * @return the quoted String (e.g. "'myString'"),
+        * or {@code null} if the input was {@code null}
+        */
+       public static String quote(String str) {
+               return (str != null ? "'" + str + "'" : null);
+       }
+
+       /**
+        * Turn the given Object into a String with single quotes
+        * if it is a String; keeping the Object as-is else.
+        * @param obj the input Object (e.g. "myString")
+        * @return the quoted String (e.g. "'myString'"),
+        * or the input object as-is if not a String
+        */
+       public static Object quoteIfString(Object obj) {
+               return (obj instanceof String ? quote((String) obj) : obj);
+       }
+
+       /**
+        * Unqualify a string qualified by a '.' dot character. For example,
+        * "this.name.is.qualified", returns "qualified".
+        * @param qualifiedName the qualified name
+        */
+       public static String unqualify(String qualifiedName) {
+               return unqualify(qualifiedName, '.');
+       }
+
+       /**
+        * Unqualify a string qualified by a separator character. For example,
+        * "this:name:is:qualified" returns "qualified" if using a ':' separator.
+        * @param qualifiedName the qualified name
+        * @param separator the separator
+        */
+       public static String unqualify(String qualifiedName, char separator) {
+               return qualifiedName.substring(qualifiedName.lastIndexOf(separator) + 1);
+       }
+
+       /**
+        * Capitalize a {@code String}, changing the first letter to
+        * upper case as per {@link Character#toUpperCase(char)}.
+        * No other letters are changed.
+        * @param str the String to capitalize, may be {@code null}
+        * @return the capitalized String, {@code null} if null
+        */
+       public static String capitalize(String str) {
+               return changeFirstCharacterCase(str, true);
+       }
+
+       /**
+        * Uncapitalize a {@code String}, changing the first letter to
+        * lower case as per {@link Character#toLowerCase(char)}.
+        * No other letters are changed.
+        * @param str the String to uncapitalize, may be {@code null}
+        * @return the uncapitalized String, {@code null} if null
+        */
+       public static String uncapitalize(String str) {
+               return changeFirstCharacterCase(str, false);
+       }
+
+       private static String changeFirstCharacterCase(String str, boolean capitalize) {
+               if (str == null || str.length() == 0) {
+                       return str;
+               }
+               StringBuilder sb = new StringBuilder(str.length());
+               if (capitalize) {
+                       sb.append(Character.toUpperCase(str.charAt(0)));
+               }
+               else {
+                       sb.append(Character.toLowerCase(str.charAt(0)));
+               }
+               sb.append(str.substring(1));
+               return sb.toString();
+       }
+
+       /**
+        * Extract the filename from the given path,
+        * e.g. "mypath/myfile.txt" -> "myfile.txt".
+        * @param path the file path (may be {@code null})
+        * @return the extracted filename, or {@code null} if none
+        */
+       public static String getFilename(String path) {
+               if (path == null) {
+                       return null;
+               }
+               int separatorIndex = path.lastIndexOf(FOLDER_SEPARATOR);
+               return (separatorIndex != -1 ? path.substring(separatorIndex + 1) : path);
+       }
+
+       /**
+        * Extract the filename extension from the given path,
+        * e.g. "mypath/myfile.txt" -> "txt".
+        * @param path the file path (may be {@code null})
+        * @return the extracted filename extension, or {@code null} if none
+        */
+       public static String getFilenameExtension(String path) {
+               if (path == null) {
+                       return null;
+               }
+               int extIndex = path.lastIndexOf(EXTENSION_SEPARATOR);
+               if (extIndex == -1) {
+                       return null;
+               }
+               int folderIndex = path.lastIndexOf(FOLDER_SEPARATOR);
+               if (folderIndex > extIndex) {
+                       return null;
+               }
+               return path.substring(extIndex + 1);
+       }
+
+       /**
+        * Strip the filename extension from the given path,
+        * e.g. "mypath/myfile.txt" -> "mypath/myfile".
+        * @param path the file path (may be {@code null})
+        * @return the path with stripped filename extension,
+        * or {@code null} if none
+        */
+       public static String stripFilenameExtension(String path) {
+               if (path == null) {
+                       return null;
+               }
+               int extIndex = path.lastIndexOf(EXTENSION_SEPARATOR);
+               if (extIndex == -1) {
+                       return path;
+               }
+               int folderIndex = path.lastIndexOf(FOLDER_SEPARATOR);
+               if (folderIndex > extIndex) {
+                       return path;
+               }
+               return path.substring(0, extIndex);
+       }
+
+       /**
+        * Apply the given relative path to the given path,
+        * assuming standard Java folder separation (i.e. "/" separators).
+        * @param path the path to start from (usually a full file path)
+        * @param relativePath the relative path to apply
+        * (relative to the full file path above)
+        * @return the full file path that results from applying the relative path
+        */
+       public static String applyRelativePath(String path, String relativePath) {
+               int separatorIndex = path.lastIndexOf(FOLDER_SEPARATOR);
+               if (separatorIndex != -1) {
+                       String newPath = path.substring(0, separatorIndex);
+                       if (!relativePath.startsWith(FOLDER_SEPARATOR)) {
+                               newPath += FOLDER_SEPARATOR;
+                       }
+                       return newPath + relativePath;
+               }
+               else {
+                       return relativePath;
+               }
+       }
+
+       /**
+        * Normalize the path by suppressing sequences like "path/.." and
+        * inner simple dots.
+        * <p>The result is convenient for path comparison. For other uses,
+        * notice that Windows separators ("\") are replaced by simple slashes.
+        * @param path the original path
+        * @return the normalized path
+        */
+       public static String cleanPath(String path) {
+               if (path == null) {
+                       return null;
+               }
+               String pathToUse = replace(path, WINDOWS_FOLDER_SEPARATOR, FOLDER_SEPARATOR);
+
+               // Strip prefix from path to analyze, to not treat it as part of the
+               // first path element. This is necessary to correctly parse paths like
+               // "file:core/../core/io/Resource.class", where the ".." should just
+               // strip the first "core" directory while keeping the "file:" prefix.
+               int prefixIndex = pathToUse.indexOf(":");
+               String prefix = "";
+               if (prefixIndex != -1) {
+                       prefix = pathToUse.substring(0, prefixIndex + 1);
+                       if (prefix.contains("/")) {
+                               prefix = "";
+                       }
+                       else {
+                               pathToUse = pathToUse.substring(prefixIndex + 1);
+                       }
+               }
+               if (pathToUse.startsWith(FOLDER_SEPARATOR)) {
+                       prefix = prefix + FOLDER_SEPARATOR;
+                       pathToUse = pathToUse.substring(1);
+               }
+
+               String[] pathArray = delimitedListToStringArray(pathToUse, FOLDER_SEPARATOR);
+               List<String> pathElements = new LinkedList<String>();
+               int tops = 0;
+
+               for (int i = pathArray.length - 1; i >= 0; i--) {
+                       String element = pathArray[i];
+                       if (CURRENT_PATH.equals(element)) {
+                               // Points to current directory - drop it.
+                       }
+                       else if (TOP_PATH.equals(element)) {
+                               // Registering top path found.
+                               tops++;
+                       }
+                       else {
+                               if (tops > 0) {
+                                       // Merging path element with element corresponding to top path.
+                                       tops--;
+                               }
+                               else {
+                                       // Normal path element found.
+                                       pathElements.add(0, element);
+                               }
+                       }
+               }
+
+               // Remaining top paths need to be retained.
+               for (int i = 0; i < tops; i++) {
+                       pathElements.add(0, TOP_PATH);
+               }
+
+               return prefix + collectionToDelimitedString(pathElements, FOLDER_SEPARATOR);
+       }
+
+       /**
+        * Compare two paths after normalization of them.
+        * @param path1 first path for comparison
+        * @param path2 second path for comparison
+        * @return whether the two paths are equivalent after normalization
+        */
+       public static boolean pathEquals(String path1, String path2) {
+               return cleanPath(path1).equals(cleanPath(path2));
+       }
+
+       /**
+        * Parse the given {@code localeString} value into a {@link Locale}.
+        * <p>This is the inverse operation of {@link Locale#toString Locale's toString}.
+        * @param localeString the locale String, following {@code Locale's}
+        * {@code toString()} format ("en", "en_UK", etc);
+        * also accepts spaces as separators, as an alternative to underscores
+        * @return a corresponding {@code Locale} instance
+        * @throws IllegalArgumentException in case of an invalid locale specification
+        */
+       public static Locale parseLocaleString(String localeString) {
+               String[] parts = tokenizeToStringArray(localeString, "_ ", false, false);
+               String language = (parts.length > 0 ? parts[0] : "");
+               String country = (parts.length > 1 ? parts[1] : "");
+               validateLocalePart(language);
+               validateLocalePart(country);
+               String variant = "";
+               if (parts.length > 2) {
+                       // There is definitely a variant, and it is everything after the country
+                       // code sans the separator between the country code and the variant.
+                       int endIndexOfCountryCode = localeString.indexOf(country, language.length()) + country.length();
+                       // Strip off any leading '_' and whitespace, what's left is the variant.
+                       variant = trimLeadingWhitespace(localeString.substring(endIndexOfCountryCode));
+                       if (variant.startsWith("_")) {
+                               variant = trimLeadingCharacter(variant, '_');
+                       }
+               }
+               return (language.length() > 0 ? new Locale(language, country, variant) : null);
+       }
+
+       private static void validateLocalePart(String localePart) {
+               for (int i = 0; i < localePart.length(); i++) {
+                       char ch = localePart.charAt(i);
+                       if (ch != '_' && ch != ' ' && !Character.isLetterOrDigit(ch)) {
+                               throw new IllegalArgumentException(
+                                               "Locale part \"" + localePart + "\" contains invalid characters");
+                       }
+               }
+       }
+
+       /**
+        * Determine the RFC 3066 compliant language tag,
+        * as used for the HTTP "Accept-Language" header.
+        * @param locale the Locale to transform to a language tag
+        * @return the RFC 3066 compliant language tag as String
+        */
+       public static String toLanguageTag(Locale locale) {
+               return locale.getLanguage() + (hasText(locale.getCountry()) ? "-" + locale.getCountry() : "");
+       }
+
+       /**
+        * Parse the given {@code timeZoneString} value into a {@link TimeZone}.
+        * @param timeZoneString the time zone String, following {@link TimeZone#getTimeZone(String)}
+        * but throwing {@link IllegalArgumentException} in case of an invalid time zone specification
+        * @return a corresponding {@link TimeZone} instance
+        * @throws IllegalArgumentException in case of an invalid time zone specification
+        */
+       public static TimeZone parseTimeZoneString(String timeZoneString) {
+               TimeZone timeZone = TimeZone.getTimeZone(timeZoneString);
+               if ("GMT".equals(timeZone.getID()) && !timeZoneString.startsWith("GMT")) {
+                       // We don't want that GMT fallback...
+                       throw new IllegalArgumentException("Invalid time zone specification '" + timeZoneString + "'");
+               }
+               return timeZone;
+       }
+
+
+       //---------------------------------------------------------------------
+       // Convenience methods for working with String arrays
+       //---------------------------------------------------------------------
+
+       /**
+        * Append the given String to the given String array, returning a new array
+        * consisting of the input array contents plus the given String.
+        * @param array the array to append to (can be {@code null})
+        * @param str the String to append
+        * @return the new array (never {@code null})
+        */
+       public static String[] addStringToArray(String[] array, String str) {
+               if (ObjectUtils.isEmpty(array)) {
+                       return new String[] {str};
+               }
+               String[] newArr = new String[array.length + 1];
+               System.arraycopy(array, 0, newArr, 0, array.length);
+               newArr[array.length] = str;
+               return newArr;
+       }
+
+       /**
+        * Concatenate the given String arrays into one,
+        * with overlapping array elements included twice.
+        * <p>The order of elements in the original arrays is preserved.
+        * @param array1 the first array (can be {@code null})
+        * @param array2 the second array (can be {@code null})
+        * @return the new array ({@code null} if both given arrays were {@code null})
+        */
+       public static String[] concatenateStringArrays(String[] array1, String[] array2) {
+               if (ObjectUtils.isEmpty(array1)) {
+                       return array2;
+               }
+               if (ObjectUtils.isEmpty(array2)) {
+                       return array1;
+               }
+               String[] newArr = new String[array1.length + array2.length];
+               System.arraycopy(array1, 0, newArr, 0, array1.length);
+               System.arraycopy(array2, 0, newArr, array1.length, array2.length);
+               return newArr;
+       }
+
+       /**
+        * Merge the given String arrays into one, with overlapping
+        * array elements only included once.
+        * <p>The order of elements in the original arrays is preserved
+        * (with the exception of overlapping elements, which are only
+        * included on their first occurrence).
+        * @param array1 the first array (can be {@code null})
+        * @param array2 the second array (can be {@code null})
+        * @return the new array ({@code null} if both given arrays were {@code null})
+        */
+       public static String[] mergeStringArrays(String[] array1, String[] array2) {
+               if (ObjectUtils.isEmpty(array1)) {
+                       return array2;
+               }
+               if (ObjectUtils.isEmpty(array2)) {
+                       return array1;
+               }
+               List<String> result = new ArrayList<String>();
+               result.addAll(Arrays.asList(array1));
+               for (String str : array2) {
+                       if (!result.contains(str)) {
+                               result.add(str);
+                       }
+               }
+               return toStringArray(result);
+       }
+
+       /**
+        * Turn given source String array into sorted array.
+        * @param array the source array
+        * @return the sorted array (never {@code null})
+        */
+       public static String[] sortStringArray(String[] array) {
+               if (ObjectUtils.isEmpty(array)) {
+                       return new String[0];
+               }
+               Arrays.sort(array);
+               return array;
+       }
+
+       /**
+        * Copy the given Collection into a String array.
+        * The Collection must contain String elements only.
+        * @param collection the Collection to copy
+        * @return the String array ({@code null} if the passed-in
+        * Collection was {@code null})
+        */
+       public static String[] toStringArray(Collection<String> collection) {
+               if (collection == null) {
+                       return null;
+               }
+               return collection.toArray(new String[collection.size()]);
+       }
+
+       /**
+        * Copy the given Enumeration into a String array.
+        * The Enumeration must contain String elements only.
+        * @param enumeration the Enumeration to copy
+        * @return the String array ({@code null} if the passed-in
+        * Enumeration was {@code null})
+        */
+       public static String[] toStringArray(Enumeration<String> enumeration) {
+               if (enumeration == null) {
+                       return null;
+               }
+               List<String> list = Collections.list(enumeration);
+               return list.toArray(new String[list.size()]);
+       }
+
+       /**
+        * Trim the elements of the given String array,
+        * calling {@code String.trim()} on each of them.
+        * @param array the original String array
+        * @return the resulting array (of the same size) with trimmed elements
+        */
+       public static String[] trimArrayElements(String[] array) {
+               if (ObjectUtils.isEmpty(array)) {
+                       return new String[0];
+               }
+               String[] result = new String[array.length];
+               for (int i = 0; i < array.length; i++) {
+                       String element = array[i];
+                       result[i] = (element != null ? element.trim() : null);
+               }
+               return result;
+       }
+
+       /**
+        * Remove duplicate Strings from the given array.
+        * Also sorts the array, as it uses a TreeSet.
+        * @param array the String array
+        * @return an array without duplicates, in natural sort order
+        */
+       public static String[] removeDuplicateStrings(String[] array) {
+               if (ObjectUtils.isEmpty(array)) {
+                       return array;
+               }
+               Set<String> set = new TreeSet<String>();
+               for (String element : array) {
+                       set.add(element);
+               }
+               return toStringArray(set);
+       }
+
+       /**
+        * Split a String at the first occurrence of the delimiter.
+        * Does not include the delimiter in the result.
+        * @param toSplit the string to split
+        * @param delimiter to split the string up with
+        * @return a two element array with index 0 being before the delimiter, and
+        * index 1 being after the delimiter (neither element includes the delimiter);
+        * or {@code null} if the delimiter wasn't found in the given input String
+        */
+       public static String[] split(String toSplit, String delimiter) {
+               if (!hasLength(toSplit) || !hasLength(delimiter)) {
+                       return null;
+               }
+               int offset = toSplit.indexOf(delimiter);
+               if (offset < 0) {
+                       return null;
+               }
+               String beforeDelimiter = toSplit.substring(0, offset);
+               String afterDelimiter = toSplit.substring(offset + delimiter.length());
+               return new String[] {beforeDelimiter, afterDelimiter};
+       }
+
+       /**
+        * Take an array Strings and split each element based on the given delimiter.
+        * A {@code Properties} instance is then generated, with the left of the
+        * delimiter providing the key, and the right of the delimiter providing the value.
+        * <p>Will trim both the key and value before adding them to the
+        * {@code Properties} instance.
+        * @param array the array to process
+        * @param delimiter to split each element using (typically the equals symbol)
+        * @return a {@code Properties} instance representing the array contents,
+        * or {@code null} if the array to process was null or empty
+        */
+       public static Properties splitArrayElementsIntoProperties(String[] array, String delimiter) {
+               return splitArrayElementsIntoProperties(array, delimiter, null);
+       }
+
+       /**
+        * Take an array Strings and split each element based on the given delimiter.
+        * A {@code Properties} instance is then generated, with the left of the
+        * delimiter providing the key, and the right of the delimiter providing the value.
+        * <p>Will trim both the key and value before adding them to the
+        * {@code Properties} instance.
+        * @param array the array to process
+        * @param delimiter to split each element using (typically the equals symbol)
+        * @param charsToDelete one or more characters to remove from each element
+        * prior to attempting the split operation (typically the quotation mark
+        * symbol), or {@code null} if no removal should occur
+        * @return a {@code Properties} instance representing the array contents,
+        * or {@code null} if the array to process was {@code null} or empty
+        */
+       public static Properties splitArrayElementsIntoProperties(
+                       String[] array, String delimiter, String charsToDelete) {
+
+               if (ObjectUtils.isEmpty(array)) {
+                       return null;
+               }
+               Properties result = new Properties();
+               for (String element : array) {
+                       if (charsToDelete != null) {
+                               element = deleteAny(element, charsToDelete);
+                       }
+                       String[] splittedElement = split(element, delimiter);
+                       if (splittedElement == null) {
+                               continue;
+                       }
+                       result.setProperty(splittedElement[0].trim(), splittedElement[1].trim());
+               }
+               return result;
+       }
+
+       /**
+        * Tokenize the given String into a String array via a StringTokenizer.
+        * Trims tokens and omits empty tokens.
+        * <p>The given delimiters string is supposed to consist of any number of
+        * delimiter characters. Each of those characters can be used to separate
+        * tokens. A delimiter is always a single character; for multi-character
+        * delimiters, consider using {@code delimitedListToStringArray}
+        * @param str the String to tokenize
+        * @param delimiters the delimiter characters, assembled as String
+        * (each of those characters is individually considered as delimiter).
+        * @return an array of the tokens
+        * @see java.util.StringTokenizer
+        * @see String#trim()
+        * @see #delimitedListToStringArray
+        */
+       public static String[] tokenizeToStringArray(String str, String delimiters) {
+               return tokenizeToStringArray(str, delimiters, true, true);
+       }
+
+       /**
+        * Tokenize the given String into a String array via a StringTokenizer.
+        * <p>The given delimiters string is supposed to consist of any number of
+        * delimiter characters. Each of those characters can be used to separate
+        * tokens. A delimiter is always a single character; for multi-character
+        * delimiters, consider using {@code delimitedListToStringArray}
+        * @param str the String to tokenize
+        * @param delimiters the delimiter characters, assembled as String
+        * (each of those characters is individually considered as delimiter)
+        * @param trimTokens trim the tokens via String's {@code trim}
+        * @param ignoreEmptyTokens omit empty tokens from the result array
+        * (only applies to tokens that are empty after trimming; StringTokenizer
+        * will not consider subsequent delimiters as token in the first place).
+        * @return an array of the tokens ({@code null} if the input String
+        * was {@code null})
+        * @see java.util.StringTokenizer
+        * @see String#trim()
+        * @see #delimitedListToStringArray
+        */
+       public static String[] tokenizeToStringArray(
+                       String str, String delimiters, boolean trimTokens, boolean ignoreEmptyTokens) {
+
+               if (str == null) {
+                       return null;
+               }
+               StringTokenizer st = new StringTokenizer(str, delimiters);
+               List<String> tokens = new ArrayList<String>();
+               while (st.hasMoreTokens()) {
+                       String token = st.nextToken();
+                       if (trimTokens) {
+                               token = token.trim();
+                       }
+                       if (!ignoreEmptyTokens || token.length() > 0) {
+                               tokens.add(token);
+                       }
+               }
+               return toStringArray(tokens);
+       }
+
+       /**
+        * Take a String which is a delimited list and convert it to a String array.
+        * <p>A single delimiter can consists of more than one character: It will still
+        * be considered as single delimiter string, rather than as bunch of potential
+        * delimiter characters - in contrast to {@code tokenizeToStringArray}.
+        * @param str the input String
+        * @param delimiter the delimiter between elements (this is a single delimiter,
+        * rather than a bunch individual delimiter characters)
+        * @return an array of the tokens in the list
+        * @see #tokenizeToStringArray
+        */
+       public static String[] delimitedListToStringArray(String str, String delimiter) {
+               return delimitedListToStringArray(str, delimiter, null);
+       }
+
+       /**
+        * Take a String which is a delimited list and convert it to a String array.
+        * <p>A single delimiter can consists of more than one character: It will still
+        * be considered as single delimiter string, rather than as bunch of potential
+        * delimiter characters - in contrast to {@code tokenizeToStringArray}.
+        * @param str the input String
+        * @param delimiter the delimiter between elements (this is a single delimiter,
+        * rather than a bunch individual delimiter characters)
+        * @param charsToDelete a set of characters to delete. Useful for deleting unwanted
+        * line breaks: e.g. "\r\n\f" will delete all new lines and line feeds in a String.
+        * @return an array of the tokens in the list
+        * @see #tokenizeToStringArray
+        */
+       public static String[] delimitedListToStringArray(String str, String delimiter, String charsToDelete) {
+               if (str == null) {
+                       return new String[0];
+               }
+               if (delimiter == null) {
+                       return new String[] {str};
+               }
+               List<String> result = new ArrayList<String>();
+               if ("".equals(delimiter)) {
+                       for (int i = 0; i < str.length(); i++) {
+                               result.add(deleteAny(str.substring(i, i + 1), charsToDelete));
+                       }
+               }
+               else {
+                       int pos = 0;
+                       int delPos;
+                       while ((delPos = str.indexOf(delimiter, pos)) != -1) {
+                               result.add(deleteAny(str.substring(pos, delPos), charsToDelete));
+                               pos = delPos + delimiter.length();
+                       }
+                       if (str.length() > 0 && pos <= str.length()) {
+                               // Add rest of String, but not in case of empty input.
+                               result.add(deleteAny(str.substring(pos), charsToDelete));
+                       }
+               }
+               return toStringArray(result);
+       }
+
+       /**
+        * Convert a CSV list into an array of Strings.
+        * @param str the input String
+        * @return an array of Strings, or the empty array in case of empty input
+        */
+       public static String[] commaDelimitedListToStringArray(String str) {
+               return delimitedListToStringArray(str, ",");
+       }
+
+       /**
+        * Convenience method to convert a CSV string list to a set.
+        * Note that this will suppress duplicates.
+        * @param str the input String
+        * @return a Set of String entries in the list
+        */
+       public static Set<String> commaDelimitedListToSet(String str) {
+               Set<String> set = new TreeSet<String>();
+               String[] tokens = commaDelimitedListToStringArray(str);
+               for (String token : tokens) {
+                       set.add(token);
+               }
+               return set;
+       }
+
+       /**
+        * Convenience method to return a Collection as a delimited (e.g. CSV)
+        * String. E.g. useful for {@code toString()} implementations.
+        * @param coll the Collection to display
+        * @param delim the delimiter to use (probably a ",")
+        * @param prefix the String to start each element with
+        * @param suffix the String to end each element with
+        * @return the delimited String
+        */
+       public static String collectionToDelimitedString(Collection<?> coll, String delim, String prefix, String suffix) {
+               if (CollectionUtils.isEmpty(coll)) {
+                       return "";
+               }
+               StringBuilder sb = new StringBuilder();
+               Iterator<?> it = coll.iterator();
+               while (it.hasNext()) {
+                       sb.append(prefix).append(it.next()).append(suffix);
+                       if (it.hasNext()) {
+                               sb.append(delim);
+                       }
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Convenience method to return a Collection as a delimited (e.g. CSV)
+        * String. E.g. useful for {@code toString()} implementations.
+        * @param coll the Collection to display
+        * @param delim the delimiter to use (probably a ",")
+        * @return the delimited String
+        */
+       public static String collectionToDelimitedString(Collection<?> coll, String delim) {
+               return collectionToDelimitedString(coll, delim, "", "");
+       }
+
+       /**
+        * Convenience method to return a Collection as a CSV String.
+        * E.g. useful for {@code toString()} implementations.
+        * @param coll the Collection to display
+        * @return the delimited String
+        */
+       public static String collectionToCommaDelimitedString(Collection<?> coll) {
+               return collectionToDelimitedString(coll, ",");
+       }
+
+       /**
+        * Convenience method to return a String array as a delimited (e.g. CSV)
+        * String. E.g. useful for {@code toString()} implementations.
+        * @param arr the array to display
+        * @param delim the delimiter to use (probably a ",")
+        * @return the delimited String
+        */
+       public static String arrayToDelimitedString(Object[] arr, String delim) {
+               if (ObjectUtils.isEmpty(arr)) {
+                       return "";
+               }
+               if (arr.length == 1) {
+                       return ObjectUtils.nullSafeToString(arr[0]);
+               }
+               StringBuilder sb = new StringBuilder();
+               for (int i = 0; i < arr.length; i++) {
+                       if (i > 0) {
+                               sb.append(delim);
+                       }
+                       sb.append(arr[i]);
+               }
+               return sb.toString();
+       }
+
+       /**
+        * Convenience method to return a String array as a CSV String.
+        * E.g. useful for {@code toString()} implementations.
+        * @param arr the array to display
+        * @return the delimited String
+        */
+       public static String arrayToCommaDelimitedString(Object[] arr) {
+               return arrayToDelimitedString(arr, ",");
+       }
+
+}
+
index eea86be..01406f1 100644 (file)
@@ -48,7 +48,7 @@ public class AccountDao {
                        final Map<String, String> row = results.get(0);
                        
                        account = new Account(row.get(CODE), row.get(NAME),
-                                       row.get(SURNAME), row.get(PASSWORD), row.get(APP_ROLE_CODE));
+                                       row.get(SURNAME), null, row.get(APP_ROLE_CODE));
                }
                
                return account;
@@ -112,7 +112,7 @@ public class AccountDao {
                        final Map<String, String> row = results.get(0);
                        
                        account = new Account(row.get(CODE), row.get(NAME),
-                                       row.get(SURNAME), row.get(PASSWORD), row.get(APP_ROLE_CODE));
+                                       row.get(SURNAME), null, row.get(APP_ROLE_CODE));
                }
                
                return account;
index c163d5d..657bebd 100644 (file)
@@ -3,43 +3,64 @@ package com.prueba.resources.controllers;
 import java.io.ByteArrayOutputStream;
 import java.io.IOException;
 import java.io.InputStream;
+import java.io.OutputStream;
 import java.nio.charset.Charset;
+import java.util.Map;
 
 import com.fasterxml.jackson.databind.ObjectMapper;
 import com.prueba.api.persistence.Account;
+import com.prueba.core.context.util.AntPathMatcher;
 import com.prueba.core.web.controller.Controller;
 import com.prueba.services.impl.ApiServiceImpl;
 import com.sun.net.httpserver.HttpExchange;
 
 public class ApiController implements Controller {
+       private static final String USER_NAME_PARAM = "username";
+       private static final String API_URL_PATTERN = "/app/api/{" + USER_NAME_PARAM + "}";
 
        @Override
        public void handle(HttpExchange httpExchange) throws IOException {
-               String requestMethod = httpExchange.getRequestMethod();
-               ApiServiceImpl apiService = new ApiServiceImpl();
-
-               String body = getBody(httpExchange);
-               
-               ObjectMapper mapper = new ObjectMapper();
-               Account account = mapper.readValue(body, Account.class);
+               final String requestMethod = httpExchange.getRequestMethod();
+               final ApiServiceImpl apiService = new ApiServiceImpl();
+               final ObjectMapper mapper = new ObjectMapper();
+       
+               final String uri = httpExchange.getRequestURI().toString();
+               final AntPathMatcher pathMatcher = new AntPathMatcher();
+               final Map<String, String> variables = pathMatcher.extractUriTemplateVariables(API_URL_PATTERN, uri);
+               final String userNameParam = variables.get(USER_NAME_PARAM);
                
                switch (requestMethod) {
-               case "GET":
-                       apiService.findAccountByCode("GUMARTIN");
-                       break;
-               case "POST":
-                       apiService.createAccount(account);
-                       break;
-               case "DELETE":
-                       apiService.deleteAccountByCode("GUMARTIN");
-                       break;
-               default:
-                       httpExchange.sendResponseHeaders(404, 0);
-                       break;
-       }
-               
-               
-               
+                       case "GET":
+                               Account account = apiService.findAccountByCode(userNameParam);
+                               String bodyResponse = "";
+                               if (account != null) {
+                                       bodyResponse = mapper.writeValueAsString(account);
+                               }
+                               
+                               httpExchange.sendResponseHeaders(200, bodyResponse.length());
+                               
+                               try (final OutputStream os = httpExchange.getResponseBody()) {
+                                       os.write(bodyResponse.getBytes());
+                               }
+                               
+                               break;
+                       case "POST":
+                               final String bodyRequest = getBody(httpExchange);
+                               final Account accountRequest = mapper.readValue(bodyRequest, Account.class);
+                               
+                               apiService.createAccount(accountRequest);
+                               
+                               httpExchange.sendResponseHeaders(200, 0);
+                               break;
+                       case "DELETE":                  
+                               apiService.deleteAccountByCode(userNameParam);
+                               
+                               httpExchange.sendResponseHeaders(200, 0);
+                               break;
+                       default:
+                               httpExchange.sendResponseHeaders(404, 0);
+                               break;
+               }
        }
 
        protected String getBody (HttpExchange httpExchange) throws IOException {
index 1423621..6594337 100644 (file)
@@ -19,9 +19,9 @@ public class LoginServiceImpl {
        private static final String COOKIE_HEADER = "Cookie";
 
        public boolean isValidUser(String username, String password) {
-               final AccountDao dao = new AccountDao();
+               final AccountDao accountDao = new AccountDao();
                
-               if (null != dao.findByCodeAndPassword(username, password)) {
+               if (null != accountDao.findByCodeAndPassword(username, password)) {
                        return true;
                } else {
                        return false;