1 // © 2017 and later: Unicode, Inc. and others. 2 // License & terms of use: http://www.unicode.org/copyright.html#License 3 package com.ibm.icu.impl.locale; 4 5 import java.io.BufferedReader; 6 import java.io.File; 7 import java.io.InputStream; 8 import java.io.InputStreamReader; 9 import java.net.URL; 10 import java.nio.charset.Charset; 11 import java.util.Arrays; 12 import java.util.Collection; 13 import java.util.Collections; 14 import java.util.HashMap; 15 import java.util.HashSet; 16 import java.util.Iterator; 17 import java.util.LinkedHashMap; 18 import java.util.LinkedHashSet; 19 import java.util.List; 20 import java.util.Map; 21 import java.util.Map.Entry; 22 import java.util.Set; 23 import java.util.TreeMap; 24 import java.util.TreeSet; 25 import java.util.regex.Matcher; 26 import java.util.regex.Pattern; 27 28 import com.ibm.icu.util.ICUException; 29 import com.ibm.icu.util.ICUUncheckedIOException; 30 31 /** 32 * Stub class to make migration easier until we get either Guava or a higher level of Java. 33 */ 34 public class XCldrStub { 35 36 public static class Multimap<K, V> { 37 private final Map<K,Set<V>> map; 38 private final Class<Set<V>> setClass; 39 40 @SuppressWarnings("unchecked") Multimap(Map<K,Set<V>> map, Class<?> setClass)41 private Multimap(Map<K,Set<V>> map, Class<?> setClass) { 42 this.map = map; 43 this.setClass = (Class<Set<V>>) (setClass != null 44 ? setClass 45 : HashSet.class); 46 } 47 @SafeVarargs 48 @SuppressWarnings("varargs") // Not supported by Eclipse, but we need this for javac putAll(K key, V... values)49 public final Multimap<K, V> putAll(K key, V... values) { 50 if (values.length != 0) { 51 createSetIfMissing(key).addAll(Arrays.asList(values)); 52 } 53 return this; 54 } putAll(K key, Collection<V> values)55 public void putAll(K key, Collection<V> values) { 56 if (!values.isEmpty()) { 57 createSetIfMissing(key).addAll(values); 58 } 59 } putAll(Collection<K> keys, V value)60 public void putAll(Collection<K> keys, V value) { 61 for (K key : keys) { 62 put(key, value); 63 } 64 } putAll(Multimap<K, V> source)65 public void putAll(Multimap<K, V> source) { 66 for (Entry<K, Set<V>> entry : source.map.entrySet()) { 67 putAll(entry.getKey(), entry.getValue()); 68 } 69 } put(K key, V value)70 public void put(K key, V value) { 71 createSetIfMissing(key).add(value); 72 } createSetIfMissing(K key)73 private Set<V> createSetIfMissing(K key) { 74 Set<V> old = map.get(key); 75 if (old == null) { 76 map.put(key, old = getInstance()); 77 } 78 return old; 79 } getInstance()80 private Set<V> getInstance() { 81 try { 82 return setClass.newInstance(); 83 } catch (Exception e) { 84 throw new ICUException(e); 85 } 86 } get(K key)87 public Set<V> get(K key) { 88 Set<V> result = map.get(key); 89 return result; // == null ? Collections.<V>emptySet() : result; 90 } keySet()91 public Set<K> keySet() { 92 return map.keySet(); 93 } asMap()94 public Map<K, Set<V>> asMap() { 95 return map; 96 } values()97 public Set<V> values() { 98 Collection<Set<V>> values = map.values(); 99 if (values.size() == 0) { 100 return Collections.<V>emptySet(); 101 } 102 Set<V> result = getInstance(); 103 for ( Set<V> valueSet : values) { 104 result.addAll(valueSet); 105 } 106 return result; 107 } size()108 public int size() { 109 return map.size(); 110 } entries()111 public Iterable<Entry<K, V>> entries() { 112 return new MultimapIterator<>(map); 113 } 114 @Override equals(Object obj)115 public boolean equals(Object obj) { 116 return this == obj || 117 (obj != null 118 && obj.getClass() == this.getClass() 119 && map.equals(((Multimap<?,?>) obj).map)); 120 } 121 122 @Override hashCode()123 public int hashCode() { 124 return map.hashCode(); 125 } 126 } 127 128 public static class Multimaps { invertFrom(Multimap<V, K> source, R target)129 public static <K, V, R extends Multimap<K, V>> R invertFrom(Multimap<V, K> source, R target) { 130 for (Entry<V, Set<K>> entry : source.asMap().entrySet()) { 131 target.putAll(entry.getValue(), entry.getKey()); 132 } 133 return target; 134 } invertFrom(Map<V, K> source, R target)135 public static <K, V, R extends Multimap<K, V>> R invertFrom(Map<V, K> source, R target) { 136 for (Entry<V, K> entry : source.entrySet()) { 137 target.put(entry.getValue(), entry.getKey()); 138 } 139 return target; 140 } 141 /** 142 * Warning, not functionally the same as Guava; only for use in invertFrom. 143 */ forMap(Map<K,V> map)144 public static <K, V> Map<K,V> forMap(Map<K,V> map) { 145 return map; 146 } 147 } 148 149 private static class MultimapIterator<K,V> implements Iterator<Entry<K,V>>, Iterable<Entry<K,V>> { 150 private final Iterator<Entry<K, Set<V>>> it1; 151 private Iterator<V> it2 = null; 152 private final ReusableEntry<K,V> entry = new ReusableEntry<>(); 153 MultimapIterator(Map<K,Set<V>> map)154 private MultimapIterator(Map<K,Set<V>> map) { 155 it1 = map.entrySet().iterator(); 156 } 157 @Override hasNext()158 public boolean hasNext() { 159 return it1.hasNext() || it2 != null && it2.hasNext(); 160 } 161 @Override next()162 public Entry<K, V> next() { 163 if (it2 != null && it2.hasNext()) { 164 entry.value = it2.next(); 165 } else { 166 Entry<K, Set<V>> e = it1.next(); 167 entry.key = e.getKey(); 168 it2 = e.getValue().iterator(); 169 } 170 return entry; 171 } 172 @Override iterator()173 public Iterator<Entry<K, V>> iterator() { 174 return this; 175 } 176 @Override remove()177 public void remove() { 178 throw new UnsupportedOperationException(); 179 } 180 } 181 182 private static class ReusableEntry<K,V> implements Entry<K,V> { 183 K key; 184 V value; 185 @Override getKey()186 public K getKey() { 187 return key; 188 } 189 @Override getValue()190 public V getValue() { 191 return value; 192 } 193 @Override setValue(V value)194 public V setValue(V value) { 195 throw new UnsupportedOperationException(); 196 } 197 } 198 199 public static class HashMultimap<K, V> extends Multimap<K, V> { HashMultimap()200 private HashMultimap() { 201 super(new HashMap<K, Set<V>>(), HashSet.class); 202 } create()203 public static <K, V> HashMultimap<K, V> create() { 204 return new HashMultimap<>(); 205 } 206 } 207 208 public static class TreeMultimap<K, V> extends Multimap<K, V> { TreeMultimap()209 private TreeMultimap() { 210 super(new TreeMap<K, Set<V>>(), TreeSet.class); 211 } create()212 public static <K, V> TreeMultimap<K, V> create() { 213 return new TreeMultimap<>(); 214 } 215 } 216 217 public static class LinkedHashMultimap<K, V> extends Multimap<K, V> { LinkedHashMultimap()218 private LinkedHashMultimap() { 219 super(new LinkedHashMap<K, Set<V>>(), LinkedHashSet.class); 220 } create()221 public static <K, V> LinkedHashMultimap<K, V> create() { 222 return new LinkedHashMultimap<>(); 223 } 224 } 225 226 227 // public static class Counter<T> implements Iterable<T>{ 228 // private Map<T,Long> data; 229 // @Override 230 // public Iterator<T> iterator() { 231 // return data.keySet().iterator(); 232 // } 233 // public long get(T s) { 234 // Long result = data.get(s); 235 // return result != null ? result : 0L; 236 // } 237 // public void add(T item, int count) { 238 // Long result = data.get(item); 239 // data.put(item, result == null ? count : result + count); 240 // } 241 // } 242 join(T[] source, String separator)243 public static <T> String join(T[] source, String separator) { 244 StringBuilder result = new StringBuilder(); 245 for (int i = 0; i < source.length; ++i) { 246 if (i != 0) result.append(separator); 247 result.append(source[i]); 248 } 249 return result.toString(); 250 } 251 join(Iterable<T> source, String separator)252 public static <T> String join(Iterable<T> source, String separator) { 253 StringBuilder result = new StringBuilder(); 254 boolean first = true; 255 for (T item : source) { 256 if (!first) result.append(separator); 257 else first = false; 258 result.append(item.toString()); 259 } 260 return result.toString(); 261 } 262 263 public static class CollectionUtilities { join(U source, String separator)264 public static <T, U extends Iterable<T>> String join(U source, String separator) { 265 return XCldrStub.join(source, separator); 266 } 267 } 268 269 public static class Joiner { 270 private final String separator; Joiner(String separator)271 private Joiner(String separator) { 272 this.separator = separator; 273 } on(String separator)274 public static final Joiner on(String separator) { 275 return new Joiner(separator); 276 } join(T[] source)277 public <T> String join(T[] source) { 278 return XCldrStub.join(source, separator); 279 } join(Iterable<T> source)280 public <T> String join(Iterable<T> source) { 281 return XCldrStub.join(source, separator); 282 } 283 } 284 285 public static class Splitter { 286 Pattern pattern; 287 boolean trimResults = false; Splitter(char c)288 public Splitter(char c) { 289 this(Pattern.compile("\\Q" + c + "\\E")); 290 } Splitter(Pattern p)291 public Splitter(Pattern p) { 292 pattern = p; 293 } on(char c)294 public static Splitter on(char c) { 295 return new Splitter(c); 296 } on(Pattern p)297 public static Splitter on(Pattern p) { 298 return new Splitter(p); 299 } splitToList(String input)300 public List<String> splitToList(String input) { 301 String[] items = pattern.split(input); 302 if (trimResults) { 303 for (int i = 0; i < items.length; ++i) { 304 items[i] = items[i].trim(); 305 } 306 } 307 return Arrays.asList(items); 308 } trimResults()309 public Splitter trimResults() { 310 trimResults = true; 311 return this; 312 } split(String input)313 public Iterable<String> split(String input) { 314 return splitToList(input); 315 } 316 } 317 318 public static class ImmutableSet { copyOf(Set<T> values)319 public static <T> Set<T> copyOf(Set<T> values) { 320 return Collections.unmodifiableSet(new LinkedHashSet<>(values)); // copy set for safety, preserve order 321 } 322 } 323 public static class ImmutableMap { copyOf(Map<K,V> values)324 public static <K,V> Map<K,V> copyOf(Map<K,V> values) { 325 return Collections.unmodifiableMap(new LinkedHashMap<>(values)); // copy set for safety, preserve order 326 } 327 } 328 public static class ImmutableMultimap { copyOf(Multimap<K,V> values)329 public static <K,V> Multimap<K,V> copyOf(Multimap<K,V> values) { 330 LinkedHashMap<K, Set<V>> temp = new LinkedHashMap<>(); // semi-deep copy, preserve order 331 for (Entry<K, Set<V>> entry : values.asMap().entrySet()) { 332 Set<V> value = entry.getValue(); 333 temp.put(entry.getKey(), value.size() == 1 334 ? Collections.singleton(value.iterator().next()) 335 : Collections.unmodifiableSet(new LinkedHashSet<>(value))); 336 } 337 return new Multimap<>(Collections.unmodifiableMap(temp), null); 338 } 339 } 340 341 public static class FileUtilities { 342 public static final Charset UTF8 = Charset.forName("utf-8"); 343 openFile(Class<?> class1, String file)344 public static BufferedReader openFile(Class<?> class1, String file) { 345 return openFile(class1, file, UTF8); 346 } 347 openFile(Class<?> class1, String file, Charset charset)348 public static BufferedReader openFile(Class<?> class1, String file, Charset charset) { 349 // URL path = null; 350 // String externalForm = null; 351 try { 352 final InputStream resourceAsStream = class1.getResourceAsStream(file); 353 if (charset == null) { 354 charset = UTF8; 355 } 356 InputStreamReader reader = new InputStreamReader(resourceAsStream, charset); 357 BufferedReader bufferedReader = new BufferedReader(reader, 1024 * 64); 358 return bufferedReader; 359 } catch (Exception e) { 360 String className = class1 == null ? null : class1.getCanonicalName(); 361 String canonicalName = null; 362 try { 363 String relativeFileName = getRelativeFileName(class1, "../util/"); 364 canonicalName = new File(relativeFileName).getCanonicalPath(); 365 } catch (Exception e1) { 366 throw new ICUUncheckedIOException("Couldn't open file: " + file + "; relative to class: " 367 + className, e); 368 } 369 throw new ICUUncheckedIOException("Couldn't open file " + file + "; in path " + canonicalName + "; relative to class: " 370 + className, e); 371 } 372 } getRelativeFileName(Class<?> class1, String filename)373 public static String getRelativeFileName(Class<?> class1, String filename) { 374 URL resource = class1 == null ? 375 FileUtilities.class.getResource(filename) : class1.getResource(filename); 376 String resourceString = resource.toString(); 377 if (resourceString.startsWith("file:")) { 378 return resourceString.substring(5); 379 } else if (resourceString.startsWith("jar:file:")) { 380 return resourceString.substring(9); 381 } else { 382 throw new ICUUncheckedIOException("File not found: " + resourceString); 383 } 384 } 385 } 386 387 static public class RegexUtilities { findMismatch(Matcher m, CharSequence s)388 public static int findMismatch(Matcher m, CharSequence s) { 389 int i; 390 for (i = 1; i < s.length(); ++i) { 391 boolean matches = m.reset(s.subSequence(0, i)).matches(); 392 if (!matches && !m.hitEnd()) { 393 break; 394 } 395 } 396 return i - 1; 397 } showMismatch(Matcher m, CharSequence s)398 public static String showMismatch(Matcher m, CharSequence s) { 399 int failPoint = findMismatch(m, s); 400 String show = s.subSequence(0, failPoint) + "☹" + s.subSequence(failPoint, s.length()); 401 return show; 402 } 403 } 404 405 public interface Predicate<T> { 406 /** 407 * Evaluates this predicate on the given argument. 408 * 409 * @param t the input argument 410 * @return {@code true} if the input argument matches the predicate, 411 * otherwise {@code false} 412 */ test(T t)413 boolean test(T t); 414 } 415 }