/* * Copyright (c) 2007, 2022, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.InputStreamReader; import java.util.*; public class CalendarTestEngine { private static File file; private static BufferedReader in; private static int testCount; private static int lineno; private static Locale locale; private static TimeZone timezone; private static boolean leniency = true; public static void main(String[] args) throws Exception { Locale loc = Locale.getDefault(); TimeZone tz = TimeZone.getDefault(); locale = loc; timezone = tz; try { for (String arg : args) { file = new File(arg); FileInputStream fis = new FileInputStream(file); System.out.println("Starting " + file.getName() + "..."); in = new BufferedReader(new InputStreamReader(fis)); testCount = lineno = 0; run(); System.out.println("Completed " + file.getName()); } } finally { Locale.setDefault(loc); TimeZone.setDefault(tz); } } private static void run() throws Exception { String line; int section = 0; Map cals = new HashMap(); CalendarAdapter calendar = null; Result result = new Result(); while ((line = in.readLine()) != null) { lineno++; line = line.trim(); // Skip blank and comment lines if (line.length() == 0 || line.charAt(0) == '#') { continue; } int comment = line.indexOf('#'); if (comment != -1) { line = line.substring(0, comment).trim(); } Scanner sc = new Scanner(line); String token = sc.next(); Symbol operation = symbol(token); if (operation == null) { throw new RuntimeException(lineno() + "wrong op? " + token); } if (operation.type == Symbol.Type.EXCEPTION) { String className = sc.next(); Class clazz = Exceptions.get(className); Exception e = result.getException(); if (!clazz.isInstance(e)) { throw new RuntimeException(lineno() + "unexpected exception: got: " + e + ", expected=" + clazz); } } Exception x = result.getException(); if (x != null) { throw new RuntimeException(lineno(result.getLineno()) + "Unexpected exception", x); } try { switch (operation.type) { case LOCALE: { String lang = sc.next(); String country = "", var = ""; if (sc.hasNext()) { country = sc.next(); if (sc.hasNext()) { var = sc.next(); } } locale = Locale.of(lang, country, var); } break; case TIMEZONE: { if (sc.hasNext()) { String id = sc.next(); timezone = TimeZone.getTimeZone(id); if (!timezone.getID().equals(id)) { System.err.printf("Warning: line %d: may get wrong time zone? " +"(specified: %s vs. actual: %s)%n", lineno, id, timezone.getID()); } } } break; case NEW: { Symbol op = symbol(sc.next()); Calendar cal = null; switch (op.type) { case INSTANCE: cal = Calendar.getInstance(timezone, locale); break; case GREGORIAN: cal = new GregorianAdapter(timezone, locale); break; default: symbolError(op); break; } cal.setLenient(leniency); calendar = new CalendarAdapter(cal); if (sc.hasNext()) { String name = sc.next(); cals.put(name.toLowerCase(Locale.ROOT), calendar); if (!leniency) { System.out.printf("%s%s is non-lenient%n", lineno(), name); } } else { throw new RuntimeException(lineno() + "Missing associated name"); } } break; case TEST: testCount++; if (sc.hasNext()) { System.out.printf("Test#%d:%s%n", testCount, sc.findInLine(".+")); } else { System.out.printf("Test#%d:%n", testCount); } break; case USE: { String name = sc.next().toLowerCase(Locale.ROOT); CalendarAdapter c = cals.get(name); if (c == null) { throw new CalendarTestException(lineno() + "calendar " + name + " not found."); } calendar = c; } break; case ASSIGN: { long v = getLong(sc); String to = sc.next().toLowerCase(Locale.ROOT); boolean assign = true; if (sc.hasNext()) { Symbol condition = symbol(sc.next()); if (condition.type == Symbol.Type.IF) { long v1 = getLong(sc); Symbol op = symbol(sc.next()); long v2 = getLong(sc); assign = relation(v1, op, v2); } else { symbolError(condition); } } if (assign) Variable.newVar(to, v); } break; case EVAL: { long v1 = getLong(sc); String op = sc.next(); Symbol operator = symbol(op); if (operator == null) { throw new RuntimeException("op " + op + " invalid"); } long v2 = getLong(sc); if (operator.isArithmetic()) { long value = 0; switch (operator.type) { case PLUS: value = v1 + v2; break; case MINUS: value = v1 - v2; break; case MULTIPLY: value = v1 * v2; break; case DIVIDE: value = v1 / v2; break; case MOD: value = v1 % v2; break; default: symbolError(operator); break; } result.setValue(value); } else { if (!relation(v1, operator, v2)) { throw new RuntimeException("not " + v1 + " " + op + " " + v2); } } } break; case CLEAR: { Symbol sym = symbol(sc.next()); if (sym.type == Symbol.Type.ALL) { calendar.clearAll(); } else if (sym.type == Symbol.Type.FIELD) { int f = sym.value(); calendar.clearField(f); } else { symbolError(sym); } } break; case GET: { Symbol sym = symbol(sc.next()); switch (sym.type) { case FIELD: { int f = sym.value(); int v = calendar.get(f); result.setValue(v); } break; case MILLIS: { long v = calendar.getTimeInMillis(); result.setValue(v); } break; case MINIMUM: { int f = getInt(sc); int v = calendar.getMinimum(f); result.setValue(v); } break; case GREATESTMINIMUM: { int f = getInt(sc); int v = calendar.getGreatestMinimum(f); result.setValue(v); } break; case ACTUALMINIMUM: { int f = getInt(sc); int v = calendar.getActualMinimum(f); result.setValue(v); } break; case MAXIMUM: { int f = getInt(sc); int v = calendar.getMaximum(f); result.setValue(v); } break; case LEASTMAXIMUM: { int f = getInt(sc); int v = calendar.getLeastMaximum(f); result.setValue(v); } break; case ACTUALMAXIMUM: { int f = getInt(sc); int v = calendar.getActualMaximum(f); result.setValue(v); } break; case FIRSTDAYOFWEEK: { result.setValue(calendar.getFirstDayOfWeek()); } break; case MINIMALDAYSINFIRSTWEEK: { int v = calendar.getMinimalDaysInFirstWeek(); result.setValue(v); } break; default: symbolError(sym); break; } } break; case ADD: case ROLL: { Symbol sym = symbol(sc.next()); if (sym.type == Symbol.Type.FIELD) { int f = sym.value(); int v = sc.nextInt(); switch (operation.type) { case ADD: calendar.add(f, v); break; case ROLL: calendar.roll(f, v); break; } } else { symbolError(sym); } } break; case SET: { Symbol sym = symbol(sc.next()); switch (sym.type) { case FIELD: { int f = sym.value(); int v = getInt(sc); calendar.set(f, v); } break; case MILLIS: { long v = getLong(sc); calendar.setTimeInMillis(v); } break; case DATE: { int a = getInt(sc); int b = getInt(sc); int c = getInt(sc); if (sc.hasNext()) { int d = getInt(sc); // era, year, month, dayOfMonth calendar.setDate(a, b, c, d); } else { // year, month, dayOfMonth calendar.setDate(a, b, c); } } break; case DATETIME: { int y = getInt(sc); int m = getInt(sc); int d = getInt(sc); int hh = getInt(sc); int mm = getInt(sc); int ss = getInt(sc); calendar.setDateTime(y, m, d, hh, mm, ss); } break; case TIMEOFDAY: { int hh = getInt(sc); int mm = getInt(sc); int ss = getInt(sc); int ms = getInt(sc); calendar.setTimeOfDay(hh, mm, ss, ms); } break; case FIRSTDAYOFWEEK: { int v = getInt(sc); calendar.setFirstDayOfWeek(v); } break; case MINIMALDAYSINFIRSTWEEK: { int v = getInt(sc); calendar.setMinimalDaysInFirstWeek(v); } break; case LENIENT: if (calendar != null) { calendar.setLenient(true); } leniency = true; break; case NONLENIENT: if (calendar != null) { calendar.setLenient(false); } leniency = false; break; default: symbolError(sym); } if (sc.hasNext()) { throw new RuntimeException(lineno() + "extra param(s) " + sc.findInLine(".+")); } } break; case CHECK: { Symbol sym = symbol(sc.next()); boolean stat = false; switch (sym.type) { case MILLIS: { long millis = getLong(sc); stat = calendar.checkMillis(millis); } break; case FIELD: { int f = sym.value(); int v = getInt(sc); stat = calendar.checkField(f, v); } break; case DATE: { int a = getInt(sc); int b = getInt(sc); int c = getInt(sc); if (sc.hasNext()) { int d = getInt(sc); // era year month dayOfMonth stat = calendar.checkDate(a, b, c, d); } else { // year month dayOfMonth stat = calendar.checkDate(a, b, c); } } break; case DATETIME: { int y = getInt(sc); int m = getInt(sc); int d = getInt(sc); int hh = getInt(sc); int mm = getInt(sc); int ss = getInt(sc); if (sc.hasNext()) { int ms = getInt(sc); stat = calendar.checkDateTime(y, m, d, hh, mm, ss, ms); } else { stat = calendar.checkDateTime(y, m, d, hh, mm, ss); } } break; case TIMEOFDAY: { int hh = sc.nextInt(); int mm = sc.nextInt(); int ss = sc.nextInt(); int millis = sc.nextInt(); stat = calendar.checkTimeOfDay(hh, mm, ss, millis); } break; case MINIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkMinimum(f, v); } break; case GREATESTMINIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkGreatestMinimum(f, v); } break; case ACTUALMINIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkActualMinimum(f, v); } break; case MAXIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkMaximum(f, v); } break; case LEASTMAXIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkLeastMaximum(f, v); } break; case ACTUALMAXIMUM: { int f = getInt(sc); int v = getInt(sc); stat = calendar.checkActualMaximum(f, v); } break; default: throw new RuntimeException(lineno() + "Unknown operand"); } if (!stat) { throw new RuntimeException(lineno() + calendar.getMessage()); } } break; case PRINT: { String s = sc.next(); if (s.charAt(0) == '$') { Variable var = variable(s); if (var == null) throw new RuntimeException(lineno() + "Unknown token: " + s); System.out.printf("%s%s=%d%n", lineno(), s, var.longValue()); break; } Symbol sym = symbol(s); switch (sym.type) { case INSTANCE: { Calendar cal = calendar; String name = "current"; if (sc.hasNext()) { name = sc.next(); cal = cals.get(name.toLowerCase(Locale.ROOT)); } System.out.printf("%s%s=%s%n", lineno(), name, cal); } break; case FIELD: { int f = sym.value(); String remark = ""; if (sc.hasNext()) { remark = sc.findInLine(".+"); } System.out.printf("%s%s=%d %s%n", lineno(), calendar.fieldName(f), calendar.get(f), remark); } break; case MILLIS: { String remark = ""; if (sc.hasNext()) { remark = sc.findInLine(".+"); } System.out.printf("%sMillis=%d %s%n", lineno(), calendar.getTimeInMillis(), remark); } break; case MINIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getMinimum(getInt(sc))); break; case GREATESTMINIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getGreatestMinimum(getInt(sc))); break; case ACTUALMINIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getActualMinimum(getInt(sc))); break; case MAXIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getMaximum(getInt(sc))); break; case LEASTMAXIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getLeastMaximum(getInt(sc))); break; case ACTUALMAXIMUM: System.out.printf("%s%s=%d%n", lineno(), s, calendar.getActualMaximum(getInt(sc))); break; case DATE: System.out.println(lineno() + calendar.toDateString()); break; case DATETIME: System.out.println(lineno() + calendar.toDateTimeString()); break; case TIMEZONE: System.out.println(lineno() + "timezone=" + timezone); break; case LOCALE: System.out.println(lineno() + "locale=" + locale); break; } } } } catch (CalendarTestException cte) { throw cte; } catch (NoSuchElementException nsee) { throw new NoSuchElementException(lineno() + "syntax error"); } catch (Exception e) { result.setException(e); result.setLineno(lineno); } } Exception x = result.getException(); if (x != null) { throw new RuntimeException(lineno(result.getLineno()) + "Unexpected exception", x); } } private static Symbol symbol(String s) { return Symbol.get(s.toLowerCase(Locale.ROOT)); } private static Variable variable(String s) { return Variable.get(s.toLowerCase(Locale.ROOT)); } private static int getInt(Scanner sc) { if (sc.hasNextInt()) { return sc.nextInt(); } String s = sc.next(); if (s.charAt(0) == '$') { Variable var = variable(s); if (var == null) throw new RuntimeException(lineno() + "Unknown token: " + s); return var.intValue(); } Symbol sym = symbol(s); if (sym == null) throw new RuntimeException(lineno() + "Unknown token: " + s); return sym.value(); } private static long getLong(Scanner sc) { if (sc.hasNextLong()) { return sc.nextLong(); } String s = sc.next(); if (s.charAt(0) == '$') { Variable var = variable(s); if (var == null) throw new RuntimeException(lineno() + "Unknown token: " + s); return var.longValue(); } Symbol sym = symbol(s); if (sym == null) throw new RuntimeException(lineno() + "Unknown token: " + s); return sym.value(); } private static boolean relation(long v1, Symbol relop, long v2) { boolean result = false; switch (relop.type) { case GT: result = v1 > v2; break; case GE: result = v1 >= v2; break; case EQ: result = v1 == v2; break; case NEQ: result = v1 != v2; break; case LE: result = v1 <= v2; break; case LT: result = v1 < v2; break; } return result; } private static String lineno() { return lineno(lineno); } private static String lineno(int ln) { return file.getName() + ":" + ln + ": "; } private static void symbolError(Symbol sym) { throw new RuntimeException(lineno + ": unexpected symbol: " + sym); } }