import java.util.*;
import java.util.zip.*;
import java.util.List;
import java.util.regex.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.*;
import java.util.concurrent.locks.*;
import javax.swing.*;
import javax.swing.event.*;
import javax.swing.text.*;
import javax.swing.table.*;
import java.io.*;
import java.net.*;
import java.lang.reflect.*;
import java.lang.ref.*;
import java.lang.management.*;
import java.security.*;
import java.security.spec.*;
import java.awt.*;
import java.awt.event.*;
import java.awt.image.*;
import javax.imageio.*;
import java.math.*;
class main {
static void test_parseFragmentExpressionToFunctionCall() {
  assertEqualsVerbose(new FunctionCall("hasWord", new SymVar("$x")),
    parseFragmentExpressionToFunctionCall(
      formatWithFragmentsAndDollarSymVars("hasWord($x)")));
  assertEqualsVerbose(new FunctionCall("ewic_notSame", new SymVar("$x"), "zimmer"),
    parseFragmentExpressionToFunctionCall(
      formatWithFragmentsAndDollarSymVars("ewic_notSame($x, *)", "zimmer")));
}
static  A assertEqualsVerbose(Object x, A y) {
  assertEqualsVerbose((String) null, x, y);
  return y;
}
static  A assertEqualsVerbose(String msg, Object x, A y) {
  if (!eq(x, y)) {
    
    throw fail((msg != null ? msg + ": " : "") + /*sfu*/(y) + " != " + /*sfu*/(x));
  } else
    print("OK: " + /*sfu*/(x));
  return y;
}
static FunctionCall parseFragmentExpressionToFunctionCall(Object o) {
  if (!(o instanceof List)) return null;
  List l =  (List) o;
  Object first = first(l);
  if (!(first instanceof Fragment)) return null;
  String first2 = ((Fragment) first).text;
  if (!isIdentifierPlusOpeningBracket(first2)) return null;
  if (!eq(last(l), new Fragment(")"))) return null;
  int i = 1, n = l(l)-1;
  String fname = dropLast(first2);
  List args = new ArrayList();
  while (i < n) {
    args.add(get(l, i++));
    if (i >= n) break;
    if (!eq(get(l, i++), new Fragment(","))) return null;
  }
  return new FunctionCall(fname, toObjectArray(args));
}
static Object formatWithFragmentsAndDollarSymVars(String starPattern, Object... args) {
  List tok = javaTok(starPattern);
  int iArg = 0, i = 1;
  List out = new ArrayList();
  while (i < l(tok)) {
    int j = smartIndexOfAsteriskOrDollarVar(tok, i);
    if (j > i)
      out.add(new Fragment(joinSubList(tok, i, j-1)));
    if (j < l(tok)) {
      String t = tok.get(j);
      if (eq(t, "*"))
        out.add(args[iArg++]);
      else
        out.add(new SymVar(t));
    }
    i = j+2;
  }
  return l(out) == 1 ? first(out) : out;
}
static boolean eq(Object a, Object b) {
  return a == b || (a == null ? b == null : b != null && a.equals(b));
}
static RuntimeException fail() { throw new RuntimeException("fail"); }
static RuntimeException fail(Throwable e) { throw asRuntimeException(e); }
static RuntimeException fail(Object msg) { throw new RuntimeException(String.valueOf(msg)); }
static RuntimeException fail(String msg) { throw new RuntimeException(msg == null ? "" : msg); }
static RuntimeException fail(String msg, Throwable innerException) { throw new RuntimeException(msg, innerException); }
static volatile StringBuffer local_log = new StringBuffer(); // not redirected
static volatile Appendable print_log = local_log; // might be redirected, e.g. to main bot
// in bytes - will cut to half that
static volatile int print_log_max = 1024*1024;
static volatile int local_log_max = 100*1024;
static boolean print_silent = false; // total mute if set
static Object print_byThread_lock = new Object();
static volatile ThreadLocal