1 package org.unicode.cldr.test; 2 3 import java.util.HashMap; 4 import java.util.HashSet; 5 import java.util.Map; 6 import java.util.Map.Entry; 7 import java.util.Set; 8 import java.util.TreeMap; 9 import java.util.regex.Matcher; 10 import java.util.regex.Pattern; 11 12 import org.unicode.cldr.util.CLDRFile; 13 import org.unicode.cldr.util.CLDRFile.Status; 14 import org.unicode.cldr.util.DateTimeCanonicalizer.DateTimePatternType; 15 import org.unicode.cldr.util.PatternCache; 16 17 import com.ibm.icu.text.DateTimePatternGenerator; 18 import com.ibm.icu.text.DateTimePatternGenerator.VariableField; 19 20 /** 21 * Class for computing the date order of date formats. 22 * This class is was originally package-visible, but has been modified to public 23 * for the sake of the unit test. 24 */ 25 public class DateOrder implements Comparable<DateOrder> { 26 private int etype1; 27 private int etype2; 28 DateOrder(int a, int b)29 public DateOrder(int a, int b) { 30 etype1 = a; 31 etype2 = b; 32 } 33 34 @Override equals(Object obj)35 public boolean equals(Object obj) { 36 DateOrder that = (DateOrder) obj; 37 return that.etype1 == etype1 && that.etype2 == etype2; 38 } 39 40 @Override hashCode()41 public int hashCode() { 42 return etype1 * 37 + etype2; 43 } 44 45 @Override toString()46 public String toString() { 47 return "<" + toString2(etype1) + "," + toString2(etype2) + ">"; 48 } 49 toString2(int etype)50 private String toString2(int etype) { 51 switch (etype >> 1) { 52 53 } 54 return (VariableField.getCanonicalCode(etype >> 1)) + ((etype & 1) == 0 ? "" : "ⁿ"); 55 } 56 57 @Override compareTo(DateOrder that)58 public int compareTo(DateOrder that) { 59 int diff; 60 if (0 != (diff = etype1 - that.etype1)) { 61 return diff; 62 } 63 return etype2 - that.etype2; 64 } 65 getOrderingInfo(CLDRFile plain, CLDRFile resolved, DateTimePatternGenerator.FormatParser fp)66 public static Map<String, Map<DateOrder, String>> getOrderingInfo(CLDRFile plain, CLDRFile resolved, 67 DateTimePatternGenerator.FormatParser fp) { 68 Map<String, Map<DateOrder, String>> pathsWithConflictingOrder2sample = new HashMap<>(); 69 Status status = new Status(); 70 try { 71 Map<String, Map<DateOrder, Set<String>>> type2order2set = new HashMap<>(); 72 Matcher typeMatcher = PatternCache.get("\\[@type=\"([^\"]*)\"]").matcher(""); 73 int[] soFar = new int[50]; 74 int lenSoFar = 0; 75 for (String path : resolved) { 76 if (DateTimePatternType.STOCK_AVAILABLE_INTERVAL_PATTERNS.contains(DateTimePatternType.fromPath(path))) { 77 if (path.contains("[@id=\"Ed\"]")) { 78 continue; 79 } 80 if (!path.equals(status.pathWhereFound)) { 81 continue; 82 } 83 typeMatcher.reset(path).find(); 84 String type = typeMatcher.group(1); 85 Map<DateOrder, Set<String>> pairCount = type2order2set.get(type); 86 if (pairCount == null) { 87 type2order2set.put(type, pairCount = new HashMap<>()); 88 } 89 boolean isInterval = path.contains("intervalFormatItem"); 90 lenSoFar = 0; 91 String value = resolved.getStringValue(path); 92 // register a comparison for all of the items so far 93 for (Object item : fp.set(value).getItems()) { 94 if (item instanceof VariableField) { 95 VariableField variable = (VariableField) item; 96 int eType = variable.getType() * 2 + (variable.isNumeric() ? 1 : 0); 97 if (isInterval && find(eType, soFar, lenSoFar)) { 98 lenSoFar = 0; // restart the clock 99 soFar[lenSoFar++] = eType; 100 continue; 101 } 102 for (int i = 0; i < lenSoFar; ++i) { 103 DateOrder order = new DateOrder(soFar[i], eType); 104 Set<String> paths = pairCount.get(order); 105 if (paths == null) { 106 pairCount.put(order, paths = new HashSet<>()); 107 } 108 paths.add(path); 109 } 110 soFar[lenSoFar++] = eType; 111 } 112 } 113 } 114 } 115 // determine conflicts, and mark 116 for (Entry<String, Map<DateOrder, Set<String>>> typeAndOrder2set : type2order2set.entrySet()) { 117 Map<DateOrder, Set<String>> pairCount = typeAndOrder2set.getValue(); 118 HashSet<DateOrder> alreadySeen = new HashSet<>(); 119 for (Entry<DateOrder, Set<String>> entry : pairCount.entrySet()) { 120 DateOrder thisOrder = entry.getKey(); 121 if (alreadySeen.contains(thisOrder)) { 122 continue; 123 } 124 DateOrder reverseOrder = new DateOrder(thisOrder.etype2, thisOrder.etype1); 125 Set<String> reverseSet = pairCount.get(reverseOrder); 126 DateOrder sample = thisOrder.compareTo(reverseOrder) < 0 ? thisOrder : reverseOrder; 127 128 Set<String> thisPaths = entry.getValue(); 129 if (reverseSet != null) { 130 addConflictingPaths(plain, sample, reverseSet, thisPaths, pathsWithConflictingOrder2sample); 131 addConflictingPaths(plain, sample, thisPaths, reverseSet, pathsWithConflictingOrder2sample); 132 alreadySeen.add(reverseOrder); 133 } 134 } 135 } 136 // for debugging, show conflicts 137 if (CheckDates.GREGORIAN_ONLY) { 138 for (Entry<String, Map<DateOrder, String>> entry : pathsWithConflictingOrder2sample.entrySet()) { 139 String path1 = entry.getKey(); 140 String locale1 = resolved.getSourceLocaleID(path1, status); 141 String value1 = resolved.getStringValue(path1); 142 Map<DateOrder, String> orderString = entry.getValue(); 143 for (Entry<DateOrder, String> entry2 : orderString.entrySet()) { 144 DateOrder order2 = entry2.getKey(); 145 String path2 = entry2.getValue(); 146 String locale2 = resolved.getSourceLocaleID(path2, status); 147 String value2 = resolved.getStringValue(path2); 148 System.out.println(order2 + "\t" + value1 + "\t" + value2 + "\t" + locale1 + "\t" + locale2 149 + "\t" + path1 + "\t" + path2); 150 } 151 } 152 } 153 } catch (RuntimeException e) { 154 throw e; 155 } 156 return pathsWithConflictingOrder2sample; 157 } 158 159 /** 160 * Add paths with a conflicting date order to the specified map. 161 * 162 * @param cldrFile 163 * @param order 164 * @param paths 165 * the set of paths to add conflicting paths for 166 * @param conflictingPaths 167 * the set of conflicting paths 168 * @param pathsWithConflictingOrder2sample 169 */ 170 private static void addConflictingPaths(CLDRFile cldrFile, DateOrder order, Set<String> paths, 171 Set<String> conflictingPaths, Map<String, Map<DateOrder, String>> pathsWithConflictingOrder2sample) { 172 for (String first : paths) { 173 FormatType firstType = FormatType.getType(first); 174 for (String otherPath : conflictingPaths) { 175 FormatType otherType = FormatType.getType(otherPath); 176 // Add the first conflicting path that has a high enough 177 // importance to be considered. 178 if (!otherType.isLessImportantThan(firstType)) { 179 addItem(cldrFile, first, order, otherPath, pathsWithConflictingOrder2sample); 180 break; 181 } 182 } 183 } 184 } 185 186 private static boolean find(int eType, int[] soFar, int lenSoFar) { 187 for (int i = 0; i < lenSoFar; ++i) { 188 if (eType == soFar[i]) { 189 return true; 190 } 191 } 192 return false; 193 } 194 195 private static void addItem(CLDRFile plain, String path, DateOrder sample, 196 String conflictingPath, Map<String, Map<DateOrder, String>> pathsWithConflictingOrder2sample) { 197 String value = plain.getStringValue(path); 198 if (value == null) { 199 return; 200 } 201 Map<DateOrder, String> order2path = pathsWithConflictingOrder2sample.get(path); 202 if (order2path == null) { 203 pathsWithConflictingOrder2sample.put(path, order2path = new TreeMap<>()); 204 } 205 order2path.put(sample, conflictingPath); 206 } 207 208 /** 209 * Enum for deciding the priority of paths for checking date order 210 * consistency. 211 */ 212 private enum FormatType { 213 DATE(3), TIME(3), AVAILABLE(2), INTERVAL(1); 214 private static final Pattern DATETIME_PATTERN = PatternCache.get("/(date|time|available|interval)Formats"); 215 // Types with a higher value have higher importance. 216 private int importance; 217 218 private FormatType(int importance) { 219 this.importance = importance; 220 } 221 222 /** 223 * @param path 224 * @return the format type of the specified path 225 */ 226 public static FormatType getType(String path) { 227 Matcher matcher = DATETIME_PATTERN.matcher(path); 228 if (matcher.find()) { 229 return FormatType.valueOf(matcher.group(1).toUpperCase()); 230 } 231 throw new IllegalArgumentException("Path is not a datetime format type: " + path); 232 } 233 234 /** 235 * @return true if this FormatType is of lower importance than otherType 236 */ 237 public boolean isLessImportantThan(FormatType otherType) { 238 return otherType.importance - importance > 0; 239 } 240 } 241 }