| /* | |
| * Copyright 2011 gitblit.com. | |
| * | |
| * 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.gitblit.utils; | |
| import java.io.ByteArrayOutputStream; | |
| import java.io.UnsupportedEncodingException; | |
| import java.nio.ByteBuffer; | |
| import java.nio.CharBuffer; | |
| import java.nio.charset.CharacterCodingException; | |
| import java.nio.charset.Charset; | |
| import java.nio.charset.CharsetDecoder; | |
| import java.nio.charset.IllegalCharsetNameException; | |
| import java.nio.charset.UnsupportedCharsetException; | |
| import java.security.MessageDigest; | |
| import java.security.NoSuchAlgorithmException; | |
| import java.util.ArrayList; | |
| import java.util.Arrays; | |
| import java.util.Collection; | |
| import java.util.Collections; | |
| import java.util.Comparator; | |
| import java.util.LinkedHashSet; | |
| import java.util.List; | |
| import java.util.Set; | |
| import java.util.regex.Matcher; | |
| import java.util.regex.Pattern; | |
| import java.util.regex.PatternSyntaxException; | |
| /** | |
| * Utility class of string functions. | |
| * | |
| * @author James Moger | |
| * | |
| */ | |
| public class StringUtils { | |
| public static final String MD5_TYPE = "MD5:"; | |
| public static final String COMBINED_MD5_TYPE = "CMD5:"; | |
| /** | |
| * Returns true if the string is null or empty. | |
| * | |
| * @param value | |
| * @return true if string is null or empty | |
| */ | |
| public static boolean isEmpty(String value) { | |
| return value == null || value.trim().length() == 0; | |
| } | |
| /** | |
| * Replaces carriage returns and line feeds with html line breaks. | |
| * | |
| * @param string | |
| * @return plain text with html line breaks | |
| */ | |
| public static String breakLinesForHtml(String string) { | |
| return string.replace("\r\n", "<br/>").replace("\r", "<br/>").replace("\n", "<br/>"); | |
| } | |
| /** | |
| * Prepare text for html presentation. Replace sensitive characters with | |
| * html entities. | |
| * | |
| * @param inStr | |
| * @param changeSpace | |
| * @return plain text escaped for html | |
| */ | |
| public static String escapeForHtml(String inStr, boolean changeSpace) { | |
| return escapeForHtml(inStr, changeSpace, 4); | |
| } | |
| /** | |
| * Prepare text for html presentation. Replace sensitive characters with | |
| * html entities. | |
| * | |
| * @param inStr | |
| * @param changeSpace | |
| * @param tabLength | |
| * @return plain text escaped for html | |
| */ | |
| public static String escapeForHtml(String inStr, boolean changeSpace, int tabLength) { | |
| StringBuilder retStr = new StringBuilder(); | |
| int i = 0; | |
| while (i < inStr.length()) { | |
| if (inStr.charAt(i) == '&') { | |
| retStr.append("&"); | |
| } else if (inStr.charAt(i) == '<') { | |
| retStr.append("<"); | |
| } else if (inStr.charAt(i) == '>') { | |
| retStr.append(">"); | |
| } else if (inStr.charAt(i) == '\"') { | |
| retStr.append("""); | |
| } else if (changeSpace && inStr.charAt(i) == ' ') { | |
| retStr.append(" "); | |
| } else if (changeSpace && inStr.charAt(i) == '\t') { | |
| for (int j = 0; j < tabLength; j++) { | |
| retStr.append(" "); | |
| } | |
| } else { | |
| retStr.append(inStr.charAt(i)); | |
| } | |
| i++; | |
| } | |
| return retStr.toString(); | |
| } | |
| /** | |
| * Decode html entities back into plain text characters. | |
| * | |
| * @param inStr | |
| * @return returns plain text from html | |
| */ | |
| public static String decodeFromHtml(String inStr) { | |
| return inStr.replace("&", "&").replace("<", "<").replace(">", ">") | |
| .replace(""", "\"").replace(" ", " "); | |
| } | |
| /** | |
| * Encodes a url parameter by escaping troublesome characters. | |
| * | |
| * @param inStr | |
| * @return properly escaped url | |
| */ | |
| public static String encodeURL(String inStr) { | |
| StringBuilder retStr = new StringBuilder(); | |
| int i = 0; | |
| while (i < inStr.length()) { | |
| if (inStr.charAt(i) == '/') { | |
| retStr.append("%2F"); | |
| } else if (inStr.charAt(i) == ' ') { | |
| retStr.append("%20"); | |
| } else if (inStr.charAt(i) == '&') { | |
| retStr.append("%26"); | |
| } else if (inStr.charAt(i) == '+') { | |
| retStr.append("%2B"); | |
| } else { | |
| retStr.append(inStr.charAt(i)); | |
| } | |
| i++; | |
| } | |
| return retStr.toString(); | |
| } | |
| /** | |
| * Flatten the list of strings into a single string with the specified | |
| * separator. | |
| * | |
| * @param values | |
| * @param separator | |
| * @return flattened list | |
| */ | |
| public static String flattenStrings(String[] values, String separator) { | |
| return flattenStrings(Arrays.asList(values), separator); | |
| } | |
| /** | |
| * Flatten the list of strings into a single string with a space separator. | |
| * | |
| * @param values | |
| * @return flattened list | |
| */ | |
| public static String flattenStrings(Collection<String> values) { | |
| return flattenStrings(values, " "); | |
| } | |
| /** | |
| * Flatten the list of strings into a single string with the specified | |
| * separator. | |
| * | |
| * @param values | |
| * @param separator | |
| * @return flattened list | |
| */ | |
| public static String flattenStrings(Collection<String> values, String separator) { | |
| StringBuilder sb = new StringBuilder(); | |
| for (String value : values) { | |
| sb.append(value).append(separator); | |
| } | |
| if (sb.length() > 0) { | |
| // truncate trailing separator | |
| sb.setLength(sb.length() - separator.length()); | |
| } | |
| return sb.toString().trim(); | |
| } | |
| /** | |
| * Returns a string trimmed to a maximum length with trailing ellipses. If | |
| * the string length is shorter than the max, the original string is | |
| * returned. | |
| * | |
| * @param value | |
| * @param max | |
| * @return trimmed string | |
| */ | |
| public static String trimString(String value, int max) { | |
| if (value.length() <= max) { | |
| return value; | |
| } | |
| return value.substring(0, max - 3) + "..."; | |
| } | |
| /** | |
| * Left pad a string with the specified character, if the string length is | |
| * less than the specified length. | |
| * | |
| * @param input | |
| * @param length | |
| * @param pad | |
| * @return left-padded string | |
| */ | |
| public static String leftPad(String input, int length, char pad) { | |
| if (input.length() < length) { | |
| StringBuilder sb = new StringBuilder(); | |
| for (int i = 0, len = length - input.length(); i < len; i++) { | |
| sb.append(pad); | |
| } | |
| sb.append(input); | |
| return sb.toString(); | |
| } | |
| return input; | |
| } | |
| /** | |
| * Right pad a string with the specified character, if the string length is | |
| * less then the specified length. | |
| * | |
| * @param input | |
| * @param length | |
| * @param pad | |
| * @return right-padded string | |
| */ | |
| public static String rightPad(String input, int length, char pad) { | |
| if (input.length() < length) { | |
| StringBuilder sb = new StringBuilder(); | |
| sb.append(input); | |
| for (int i = 0, len = length - input.length(); i < len; i++) { | |
| sb.append(pad); | |
| } | |
| return sb.toString(); | |
| } | |
| return input; | |
| } | |
| /** | |
| * Calculates the SHA1 of the string. | |
| * | |
| * @param text | |
| * @return sha1 of the string | |
| */ | |
| public static String getSHA1(String text) { | |
| try { | |
| byte[] bytes = text.getBytes("iso-8859-1"); | |
| return getSHA1(bytes); | |
| } catch (UnsupportedEncodingException u) { | |
| throw new RuntimeException(u); | |
| } | |
| } | |
| /** | |
| * Calculates the SHA1 of the byte array. | |
| * | |
| * @param bytes | |
| * @return sha1 of the byte array | |
| */ | |
| public static String getSHA1(byte[] bytes) { | |
| try { | |
| MessageDigest md = MessageDigest.getInstance("SHA-1"); | |
| md.update(bytes, 0, bytes.length); | |
| byte[] digest = md.digest(); | |
| return toHex(digest); | |
| } catch (NoSuchAlgorithmException t) { | |
| throw new RuntimeException(t); | |
| } | |
| } | |
| /** | |
| * Calculates the MD5 of the string. | |
| * | |
| * @param string | |
| * @return md5 of the string | |
| */ | |
| public static String getMD5(String string) { | |
| try { | |
| return getMD5(string.getBytes("iso-8859-1")); | |
| } catch (UnsupportedEncodingException u) { | |
| throw new RuntimeException(u); | |
| } | |
| } | |
| /** | |
| * Calculates the MD5 of the string. | |
| * | |
| * @param string | |
| * @return md5 of the string | |
| */ | |
| public static String getMD5(byte [] bytes) { | |
| try { | |
| MessageDigest md = MessageDigest.getInstance("MD5"); | |
| md.reset(); | |
| md.update(bytes); | |
| byte[] digest = md.digest(); | |
| return toHex(digest); | |
| } catch (NoSuchAlgorithmException t) { | |
| throw new RuntimeException(t); | |
| } | |
| } | |
| /** | |
| * Returns the hex representation of the byte array. | |
| * | |
| * @param bytes | |
| * @return byte array as hex string | |
| */ | |
| public static String toHex(byte[] bytes) { | |
| StringBuilder sb = new StringBuilder(bytes.length * 2); | |
| for (int i = 0; i < bytes.length; i++) { | |
| if ((bytes[i] & 0xff) < 0x10) { | |
| sb.append('0'); | |
| } | |
| sb.append(Long.toString(bytes[i] & 0xff, 16)); | |
| } | |
| return sb.toString(); | |
| } | |
| /** | |
| * Returns the root path of the specified path. Returns a blank string if | |
| * there is no root path. | |
| * | |
| * @param path | |
| * @return root path or blank | |
| */ | |
| public static String getRootPath(String path) { | |
| if (path.indexOf('/') > -1) { | |
| return path.substring(0, path.lastIndexOf('/')); | |
| } | |
| return ""; | |
| } | |
| /** | |
| * Returns the path remainder after subtracting the basePath from the | |
| * fullPath. | |
| * | |
| * @param basePath | |
| * @param fullPath | |
| * @return the relative path | |
| */ | |
| public static String getRelativePath(String basePath, String fullPath) { | |
| String bp = basePath.replace('\\', '/').toLowerCase(); | |
| String fp = fullPath.replace('\\', '/').toLowerCase(); | |
| if (fp.startsWith(bp)) { | |
| String relativePath = fullPath.substring(basePath.length()).replace('\\', '/'); | |
| if (relativePath.length() > 0 && relativePath.charAt(0) == '/') { | |
| relativePath = relativePath.substring(1); | |
| } | |
| return relativePath; | |
| } | |
| return fullPath; | |
| } | |
| /** | |
| * Splits the space-separated string into a list of strings. | |
| * | |
| * @param value | |
| * @return list of strings | |
| */ | |
| public static List<String> getStringsFromValue(String value) { | |
| return getStringsFromValue(value, " "); | |
| } | |
| /** | |
| * Splits the string into a list of string by the specified separator. | |
| * | |
| * @param value | |
| * @param separator | |
| * @return list of strings | |
| */ | |
| public static List<String> getStringsFromValue(String value, String separator) { | |
| List<String> strings = new ArrayList<String>(); | |
| try { | |
| String[] chunks = value.split(separator + "(?=([^\"]*\"[^\"]*\")*[^\"]*$)"); | |
| for (String chunk : chunks) { | |
| chunk = chunk.trim(); | |
| if (chunk.length() > 0) { | |
| if (chunk.charAt(0) == '"' && chunk.charAt(chunk.length() - 1) == '"') { | |
| // strip double quotes | |
| chunk = chunk.substring(1, chunk.length() - 1).trim(); | |
| } | |
| strings.add(chunk); | |
| } | |
| } | |
| } catch (PatternSyntaxException e) { | |
| throw new RuntimeException(e); | |
| } | |
| return strings; | |
| } | |
| /** | |
| * Validates that a name is composed of letters, digits, or limited other | |
| * characters. | |
| * | |
| * @param name | |
| * @return the first invalid character found or null if string is acceptable | |
| */ | |
| public static Character findInvalidCharacter(String name) { | |
| char[] validChars = { '/', '.', '_', '-', '~', '+' }; | |
| for (char c : name.toCharArray()) { | |
| if (!Character.isLetterOrDigit(c)) { | |
| boolean ok = false; | |
| for (char vc : validChars) { | |
| ok |= c == vc; | |
| } | |
| if (!ok) { | |
| return c; | |
| } | |
| } | |
| } | |
| return null; | |
| } | |
| /** | |
| * Simple fuzzy string comparison. This is a case-insensitive check. A | |
| * single wildcard * value is supported. | |
| * | |
| * @param value | |
| * @param pattern | |
| * @return true if the value matches the pattern | |
| */ | |
| public static boolean fuzzyMatch(String value, String pattern) { | |
| if (value.equalsIgnoreCase(pattern)) { | |
| return true; | |
| } | |
| if (pattern.contains("*")) { | |
| boolean prefixMatches = false; | |
| boolean suffixMatches = false; | |
| int wildcard = pattern.indexOf('*'); | |
| String prefix = pattern.substring(0, wildcard).toLowerCase(); | |
| prefixMatches = value.toLowerCase().startsWith(prefix); | |
| if (pattern.length() > (wildcard + 1)) { | |
| String suffix = pattern.substring(wildcard + 1).toLowerCase(); | |
| suffixMatches = value.toLowerCase().endsWith(suffix); | |
| return prefixMatches && suffixMatches; | |
| } | |
| return prefixMatches || suffixMatches; | |
| } | |
| return false; | |
| } | |
| /** | |
| * Compare two repository names for proper group sorting. | |
| * | |
| * @param r1 | |
| * @param r2 | |
| * @return | |
| */ | |
| public static int compareRepositoryNames(String r1, String r2) { | |
| // sort root repositories first, alphabetically | |
| // then sort grouped repositories, alphabetically | |
| r1 = r1.toLowerCase(); | |
| r2 = r2.toLowerCase(); | |
| int s1 = r1.indexOf('/'); | |
| int s2 = r2.indexOf('/'); | |
| if (s1 == -1 && s2 == -1) { | |
| // neither grouped | |
| return r1.compareTo(r2); | |
| } else if (s1 > -1 && s2 > -1) { | |
| // both grouped | |
| return r1.compareTo(r2); | |
| } else if (s1 == -1) { | |
| return -1; | |
| } else if (s2 == -1) { | |
| return 1; | |
| } | |
| return 0; | |
| } | |
| /** | |
| * Sort grouped repository names. | |
| * | |
| * @param list | |
| */ | |
| public static void sortRepositorynames(List<String> list) { | |
| Collections.sort(list, new Comparator<String>() { | |
| @Override | |
| public int compare(String o1, String o2) { | |
| return compareRepositoryNames(o1, o2); | |
| } | |
| }); | |
| } | |
| public static String getColor(String value) { | |
| int cs = 0; | |
| for (char c : getMD5(value.toLowerCase()).toCharArray()) { | |
| cs += c; | |
| } | |
| int n = (cs % 360); | |
| float hue = ((float) n) / 360; | |
| return hsvToRgb(hue, 0.90f, 0.65f); | |
| } | |
| public static String hsvToRgb(float hue, float saturation, float value) { | |
| int h = (int) (hue * 6); | |
| float f = hue * 6 - h; | |
| float p = value * (1 - saturation); | |
| float q = value * (1 - f * saturation); | |
| float t = value * (1 - (1 - f) * saturation); | |
| switch (h) { | |
| case 0: | |
| return rgbToString(value, t, p); | |
| case 1: | |
| return rgbToString(q, value, p); | |
| case 2: | |
| return rgbToString(p, value, t); | |
| case 3: | |
| return rgbToString(p, q, value); | |
| case 4: | |
| return rgbToString(t, p, value); | |
| case 5: | |
| return rgbToString(value, p, q); | |
| default: | |
| throw new RuntimeException( | |
| "Something went wrong when converting from HSV to RGB. Input was " + hue + ", " | |
| + saturation + ", " + value); | |
| } | |
| } | |
| public static String rgbToString(float r, float g, float b) { | |
| String rs = Integer.toHexString((int) (r * 256)); | |
| String gs = Integer.toHexString((int) (g * 256)); | |
| String bs = Integer.toHexString((int) (b * 256)); | |
| return "#" + rs + gs + bs; | |
| } | |
| /** | |
| * Strips a trailing ".git" from the value. | |
| * | |
| * @param value | |
| * @return a stripped value or the original value if .git is not found | |
| */ | |
| public static String stripDotGit(String value) { | |
| if (value.toLowerCase().endsWith(".git")) { | |
| return value.substring(0, value.length() - 4); | |
| } | |
| return value; | |
| } | |
| /** | |
| * Count the number of lines in a string. | |
| * | |
| * @param value | |
| * @return the line count | |
| */ | |
| public static int countLines(String value) { | |
| if (isEmpty(value)) { | |
| return 0; | |
| } | |
| return value.split("\n").length; | |
| } | |
| /** | |
| * Returns the file extension of a path. | |
| * | |
| * @param path | |
| * @return a blank string or a file extension | |
| */ | |
| public static String getFileExtension(String path) { | |
| int lastDot = path.lastIndexOf('.'); | |
| if (lastDot > -1) { | |
| return path.substring(lastDot + 1); | |
| } | |
| return ""; | |
| } | |
| /** | |
| * Returns the file extension of a path. | |
| * | |
| * @param path | |
| * @return a blank string or a file extension | |
| */ | |
| public static String stripFileExtension(String path) { | |
| int lastDot = path.lastIndexOf('.'); | |
| if (lastDot > -1) { | |
| return path.substring(0, lastDot); | |
| } | |
| return path; | |
| } | |
| /** | |
| * Replace all occurences of a substring within a string with | |
| * another string. | |
| * | |
| * From Spring StringUtils. | |
| * | |
| * @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) { | |
| 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(); | |
| } | |
| /** | |
| * Decodes a string by trying several charsets until one does not throw a | |
| * coding exception. Last resort is to interpret as UTF-8 with illegal | |
| * character substitution. | |
| * | |
| * @param content | |
| * @param charsets optional | |
| * @return a string | |
| */ | |
| public static String decodeString(byte [] content, String... charsets) { | |
| Set<String> sets = new LinkedHashSet<String>(); | |
| if (!ArrayUtils.isEmpty(charsets)) { | |
| sets.addAll(Arrays.asList(charsets)); | |
| } | |
| String value = null; | |
| sets.addAll(Arrays.asList("UTF-8", "ISO-8859-1", Charset.defaultCharset().name())); | |
| for (String charset : sets) { | |
| try { | |
| Charset cs = Charset.forName(charset); | |
| CharsetDecoder decoder = cs.newDecoder(); | |
| CharBuffer buffer = decoder.decode(ByteBuffer.wrap(content)); | |
| value = buffer.toString(); | |
| break; | |
| } catch (CharacterCodingException e) { | |
| // ignore and advance to the next charset | |
| } catch (IllegalCharsetNameException e) { | |
| // ignore illegal charset names | |
| } catch (UnsupportedCharsetException e) { | |
| // ignore unsupported charsets | |
| } | |
| } | |
| if (value != null && value.startsWith("\uFEFF")) { | |
| // strip UTF-8 BOM | |
| return value.substring(1); | |
| } | |
| return value; | |
| } | |
| /** | |
| * Attempt to extract a repository name from a given url using regular | |
| * expressions. If no match is made, then return whatever trails after | |
| * the final / character. | |
| * | |
| * @param regexUrls | |
| * @return a repository path | |
| */ | |
| public static String extractRepositoryPath(String url, String... urlpatterns) { | |
| for (String urlPattern : urlpatterns) { | |
| Pattern p = Pattern.compile(urlPattern); | |
| Matcher m = p.matcher(url); | |
| while (m.find()) { | |
| String repositoryPath = m.group(1); | |
| return repositoryPath; | |
| } | |
| } | |
| // last resort | |
| if (url.lastIndexOf('/') > -1) { | |
| return url.substring(url.lastIndexOf('/') + 1); | |
| } | |
| return url; | |
| } | |
| /** | |
| * Converts a string with \nnn sequences into a UTF-8 encoded string. | |
| * @param input | |
| * @return | |
| */ | |
| public static String convertOctal(String input) { | |
| try { | |
| ByteArrayOutputStream bytes = new ByteArrayOutputStream(); | |
| Pattern p = Pattern.compile("(\\\\\\d{3})"); | |
| Matcher m = p.matcher(input); | |
| int i = 0; | |
| while (m.find()) { | |
| bytes.write(input.substring(i, m.start()).getBytes("UTF-8")); | |
| // replace octal encoded value | |
| // strip leading \ character | |
| String oct = m.group().substring(1); | |
| bytes.write(Integer.parseInt(oct, 8)); | |
| i = m.end(); | |
| } | |
| if (bytes.size() == 0) { | |
| // no octal matches | |
| return input; | |
| } else { | |
| if (i < input.length()) { | |
| // add remainder of string | |
| bytes.write(input.substring(i).getBytes("UTF-8")); | |
| } | |
| } | |
| return bytes.toString("UTF-8"); | |
| } catch (Exception e) { | |
| e.printStackTrace(); | |
| } | |
| return input; | |
| } | |
| /** | |
| * Returns the first path element of a path string. If no path separator is | |
| * found in the path, an empty string is returned. | |
| * | |
| * @param path | |
| * @return the first element in the path | |
| */ | |
| public static String getFirstPathElement(String path) { | |
| if (path.indexOf('/') > -1) { | |
| return path.substring(0, path.indexOf('/')).trim(); | |
| } | |
| return ""; | |
| } | |
| /** | |
| * Returns the last path element of a path string | |
| * | |
| * @param path | |
| * @return the last element in the path | |
| */ | |
| public static String getLastPathElement(String path) { | |
| if (path.indexOf('/') > -1) { | |
| return path.substring(path.lastIndexOf('/') + 1); | |
| } | |
| return path; | |
| } | |
| /** | |
| * Variation of String.matches() which disregards case issues. | |
| * | |
| * @param regex | |
| * @param input | |
| * @return true if the pattern matches | |
| */ | |
| public static boolean matchesIgnoreCase(String input, String regex) { | |
| Pattern p = Pattern.compile(regex, Pattern.CASE_INSENSITIVE); | |
| Matcher m = p.matcher(input); | |
| return m.matches(); | |
| } | |
| /** | |
| * Removes new line and carriage return chars from a string. | |
| * If input value is null an empty string is returned. | |
| * | |
| * @param input | |
| * @return a sanitized or empty string | |
| */ | |
| public static String removeNewlines(String input) { | |
| if (input == null) { | |
| return ""; | |
| } | |
| return input.replace('\n',' ').replace('\r', ' ').trim(); | |
| } | |
| /** | |
| * Encode the username for user in an url. | |
| * | |
| * @param name | |
| * @return the encoded name | |
| */ | |
| public static String encodeUsername(String name) { | |
| return name.replace("@", "%40").replace(" ", "%20").replace("\\", "%5C"); | |
| } | |
| /** | |
| * Decode a username from an encoded url. | |
| * | |
| * @param name | |
| * @return the decoded name | |
| */ | |
| public static String decodeUsername(String name) { | |
| return name.replace("%40", "@").replace("%20", " ").replace("%5C", "\\"); | |
| } | |
| } |