| // Copyright 2014 The Bazel Authors. All rights reserved. |
| // |
| // 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.google.devtools.build.lib.util; |
| |
| import com.google.common.base.Function; |
| import com.google.common.base.Joiner; |
| import com.google.common.base.Splitter; |
| import com.google.common.collect.Iterables; |
| |
| import java.util.ArrayList; |
| import java.util.Collection; |
| import java.util.Iterator; |
| import java.util.List; |
| |
| /** |
| * Various utility methods operating on strings. |
| */ |
| public class StringUtil { |
| /** |
| * Creates a comma-separated list of words as in English. |
| * |
| * <p>Example: ["a", "b", "c"] -> "a, b or c". |
| */ |
| public static String joinEnglishList(Iterable<?> choices) { |
| return joinEnglishList(choices, "or", ""); |
| } |
| |
| /** |
| * Creates a comma-separated list of words as in English with the given last-separator. |
| * |
| * <p>Example with lastSeparator="then": ["a", "b", "c"] -> "a, b then c". |
| */ |
| public static String joinEnglishList(Iterable<?> choices, String lastSeparator) { |
| return joinEnglishList(choices, lastSeparator, ""); |
| } |
| |
| /** |
| * Creates a comma-separated list of words as in English with the given last-separator and quotes. |
| * |
| * <p>Example with lastSeparator="then", quote="'": ["a", "b", "c"] -> "'a', 'b' then 'c'". |
| */ |
| public static String joinEnglishList(Iterable<?> choices, String lastSeparator, String quote) { |
| StringBuilder buf = new StringBuilder(); |
| for (Iterator<?> ii = choices.iterator(); ii.hasNext(); ) { |
| Object choice = ii.next(); |
| if (buf.length() > 0) { |
| buf.append(ii.hasNext() ? "," : " " + lastSeparator); |
| buf.append(" "); |
| } |
| buf.append(quote).append(choice).append(quote); |
| } |
| return buf.length() == 0 ? "nothing" : buf.toString(); |
| } |
| |
| /** |
| * Split a single space-separated string into a List of values. |
| * |
| * <p>Individual values are canonicalized such that within and |
| * across calls to this method, equal values point to the same |
| * object. |
| * |
| * <p>If the input is null, return an empty list. |
| * |
| * @param in space-separated list of values, eg "value1 value2". |
| */ |
| public static List<String> splitAndInternString(String in) { |
| List<String> result = new ArrayList<>(); |
| if (in == null) { |
| return result; |
| } |
| for (String val : Splitter.on(' ').omitEmptyStrings().split(in)) { |
| // Note that splitter returns a substring(), effectively |
| // retaining the entire "in" String. Make an explicit copy here |
| // to avoid that memory pitfall. Further, because there may be |
| // many concurrent submissions that touch the same files, |
| // attempt to use a single reference for equal strings via the |
| // deduplicator. |
| result.add(StringCanonicalizer.intern(val)); |
| } |
| return result; |
| } |
| |
| /** |
| * Lists items up to a given limit, then prints how many were omitted. |
| */ |
| public static StringBuilder listItemsWithLimit(StringBuilder appendTo, int limit, |
| Collection<?> items) { |
| Preconditions.checkState(limit > 0); |
| Joiner.on(", ").appendTo(appendTo, Iterables.limit(items, limit)); |
| if (items.size() > limit) { |
| appendTo.append(" ...(omitting ") |
| .append(items.size() - limit) |
| .append(" more item(s))"); |
| } |
| return appendTo; |
| } |
| |
| /** |
| * Returns the ordinal representation of the number. |
| */ |
| public static String ordinal(int number) { |
| switch (number) { |
| case 1: |
| return "1st"; |
| case 2: |
| return "2nd"; |
| case 3: |
| return "3rd"; |
| default: |
| return number + "th"; |
| } |
| } |
| |
| /** |
| * Appends a prefix and a suffix to each of the Strings. |
| */ |
| public static Iterable<String> append(Iterable<String> values, final String prefix, |
| final String suffix) { |
| return Iterables.transform(values, new Function<String, String>() { |
| @Override |
| public String apply(String input) { |
| return prefix + input + suffix; |
| } |
| }); |
| } |
| |
| /** |
| * Indents the specified string by the given number of characters. |
| * |
| * <p>The beginning of the string before the first newline is not indented. |
| */ |
| public static String indent(String input, int depth) { |
| StringBuilder prefix = new StringBuilder(); |
| prefix.append("\n"); |
| for (int i = 0; i < depth; i++) { |
| prefix.append(" "); |
| } |
| |
| return input.replace("\n", prefix); |
| } |
| |
| /** |
| * Strips a suffix from a string. If the string does not end with the suffix, returns null. |
| */ |
| public static String stripSuffix(String input, String suffix) { |
| return input.endsWith(suffix) |
| ? input.substring(0, input.length() - suffix.length()) |
| : null; |
| } |
| |
| /** |
| * Capitalizes the first character of a string. |
| */ |
| public static String capitalize(String input) { |
| if (input.isEmpty()) { |
| return input; |
| } |
| |
| char first = input.charAt(0); |
| char capitalized = Character.toUpperCase(first); |
| return first == capitalized ? input : capitalized + input.substring(1); |
| } |
| } |