d6b4693c05
Reviewed-by: iris, joehw
783 lines
30 KiB
Java
783 lines
30 KiB
Java
/*
|
|
* 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<String, CalendarAdapter> cals = new HashMap<String, CalendarAdapter>();
|
|
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);
|
|
}
|
|
}
|