Skip to content
Snippets Groups Projects
Select Git revision
  • 005fc6225b35925097ba1df329bd31eabefbe3f1
  • master default protected
  • ia2
  • adql2.1-ia2
  • private_rows
5 results

Token.java.backup

Blame
    • gmantele's avatar
      005fc622
      [ADQL] Recompilation of the ADQLParser classes using JavaCC 6.0. · 005fc622
      gmantele authored
      Two classes have been modified after compilation:
          - ADQLParser - a simple cast for one of the automatically generated
                         constructor.
          - ParseException - the token position has been stored for better
                             syntax error messages.
      A note has been added at the top comment for both files to highlight
      the modified parts and how to restore them after re-generation.
      005fc622
      History
      [ADQL] Recompilation of the ADQLParser classes using JavaCC 6.0.
      gmantele authored
      Two classes have been modified after compilation:
          - ADQLParser - a simple cast for one of the automatically generated
                         constructor.
          - ParseException - the token position has been stored for better
                             syntax error messages.
      A note has been added at the top comment for both files to highlight
      the modified parts and how to restore them after re-generation.
    TestADQLParser.java 49.73 KiB
    package adql.parser;
    
    import static org.junit.Assert.assertEquals;
    import static org.junit.Assert.assertFalse;
    import static org.junit.Assert.assertNotNull;
    import static org.junit.Assert.assertTrue;
    import static org.junit.Assert.fail;
    
    import java.util.ArrayList;
    import java.util.Arrays;
    import java.util.Iterator;
    
    import org.junit.After;
    import org.junit.AfterClass;
    import org.junit.Before;
    import org.junit.BeforeClass;
    import org.junit.Test;
    
    import adql.db.FunctionDef;
    import adql.db.exception.UnresolvedIdentifiersException;
    import adql.db.exception.UnsupportedFeatureException;
    import adql.parser.ADQLParser.ADQLVersion;
    import adql.parser.feature.LanguageFeature;
    import adql.parser.grammar.ADQLGrammar200Constants;
    import adql.parser.grammar.ParseException;
    import adql.parser.grammar.Token;
    import adql.query.ADQLQuery;
    import adql.query.WithItem;
    import adql.query.from.ADQLJoin;
    import adql.query.from.ADQLTable;
    import adql.query.operand.StringConstant;
    import adql.query.operand.function.geometry.CircleFunction;
    import adql.query.operand.function.geometry.ContainsFunction;
    import adql.query.operand.function.geometry.PointFunction;
    import adql.query.operand.function.geometry.RegionFunction;
    import adql.query.operand.function.string.LowerFunction;
    
    public class TestADQLParser {
    
    	@BeforeClass
    	public static void setUpBeforeClass() throws Exception {
    	}
    
    	@AfterClass
    	public static void tearDownAfterClass() throws Exception {
    	}
    
    	@Before
    	public void setUp() throws Exception {
    	}
    
    	@After
    	public void tearDown() throws Exception {
    	}
    
    	@Test
    	public void testWithClause() {
    
    		// CASE: ADQL-2.0 => ERROR
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_0);
    		try {
    			parser.parseQuery("WITH foo AS (SELECT * FROM bar) SELECT * FROM foo");
    			fail("In ADQL-2.0, the WITH should not be allowed....it does not exist!");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals(" Encountered \"WITH\". Was expecting: \"SELECT\" \n" + "(HINT: \"WITH\" is not supported in ADQL v2.0, but is however a reserved word. To use it as a column/table/schema name/alias, write it between double quotes.)", ex.getMessage());
    		}
    
    		parser = new ADQLParser(ADQLVersion.V2_1);
    		try {
    
    			// CASE: Same with ADQL-2.1 => OK
    			ADQLQuery query = parser.parseQuery("WITH foo AS (SELECT * FROM bar) SELECT * FROM foo");
    			assertNotNull(query.getWith());
    			assertEquals(1, query.getWith().size());
    			WithItem item = query.getWith().get(0);
    			assertEquals("foo", item.getLabel());
    			assertFalse(item.isLabelCaseSensitive());
    			assertEquals("SELECT *\nFROM bar", item.getQuery().toADQL());
    
    			// CASE: WITH clause with a column set => OK
    			query = parser.parseQuery("WITH foo AS (SELECT col1, col2, col3 FROM bar) SELECT * FROM foo");
    			assertNotNull(query.getWith());
    			assertEquals(1, query.getWith().size());
    			item = query.getWith().get(0);
    			assertEquals("foo", item.getLabel());
    			assertFalse(item.isLabelCaseSensitive());
    			assertEquals("SELECT col1 , col2 , col3\nFROM bar", item.getQuery().toADQL());
    
    			// CASE: more than 1 WITH clause + CTE's label case sensitivity
    			query = parser.parseQuery("WITH foo AS (SELECT col1, col2, col3 FROM bar), \"Foo2\" AS (SELECT * FROM bar2) SELECT * FROM foo NATURAL JOIN \"Foo2\"");
    			assertNotNull(query.getWith());
    			assertEquals(2, query.getWith().size());
    			item = query.getWith().get(0);
    			assertEquals("foo", item.getLabel());
    			assertFalse(item.isLabelCaseSensitive());
    			assertEquals("SELECT col1 , col2 , col3\nFROM bar", item.getQuery().toADQL());
    			item = query.getWith().get(1);
    			assertEquals("Foo2", item.getLabel());
    			assertTrue(item.isLabelCaseSensitive());
    			assertEquals("SELECT *\nFROM bar2", item.getQuery().toADQL());
    
    			// CASE: WITH clause inside a WITH clause => OK
    			query = parser.parseQuery("WITH foo  AS (WITH innerFoo AS (SELECT col1, col2, col3 FROM bar) SELECT * FROM stars NATURAL JOIN innerFoo) SELECT * FROM foo");
    			assertNotNull(query.getWith());
    			assertEquals(1, query.getWith().size());
    			item = query.getWith().get(0);
    			assertEquals("foo", item.getLabel());
    			assertFalse(item.isLabelCaseSensitive());
    			assertEquals("WITH innerFoo AS (\nSELECT col1 , col2 , col3\nFROM bar\n)\nSELECT *\nFROM stars NATURAL INNER JOIN innerFoo", item.getQuery().toADQL());
    			assertNotNull(query.getWith().get(0).getQuery().getWith());
    			assertEquals(1, query.getWith().get(0).getQuery().getWith().size());
    			item = query.getWith().get(0).getQuery().getWith().get(0);
    			assertEquals("innerFoo", item.getLabel());
    			assertFalse(item.isLabelCaseSensitive());
    			assertEquals("SELECT col1 , col2 , col3\nFROM bar", item.getQuery().toADQL());
    
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error while parsing a valid query with a WITH clause! (see console for more details)");
    		}
    	}
    
    	@Test
    	public void testConstraintList() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			/* TEST: in a constraint (i.e. Constraint() in the grammar),
    			 *       avoid ambiguity between (OPERAND) and (CONSTRAINT) */
    			try {
    				parser.parseWhere("WHERE (mag + 2) < 5"); // CONSTRAINT = (OPERAND) COMP_OPERATOR OPERAND
    				parser.parseWhere("WHERE (mag < 2)");     // CONSTRAINT = (CONSTRAINT)
    			} catch(Exception ex) {
    				ex.printStackTrace();
    				fail("[ADQL-" + version + "] Unexpected error while parsing WHERE valid conditions! (see console for more details)");
    			}
    
    			// CASE: same test but this time with an incorrect function argument
    			/*
    			 * NOTE: If this expression is not correctly parsed, the raised
    			 *       error will be about CONTAINS instead of being about PI.
    			 */
    			try {
    				parser.parseWhere("WHERE CONTAINS(PI(), CIRCLE('', ra, dec)) = 1");
    				fail("PI() is not a valid argument of CONTAINS(...)!");
    			} catch(Exception ex) {
    				assertEquals(ParseException.class, ex.getClass());
    				assertTrue(ex.getMessage().startsWith(" Encountered \"PI\". Was expecting one of: \"BOX\" \"CENTROID\" \"CIRCLE\" \"POINT\" \"POLYGON\""));
    			}
    		}
    	}
    
    	@Test
    	public void testNumericFunctionParams() {
    
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_1);
    
    		/* CASE: LOWER can only take a string in parameter, but according to the
    		 *       grammar (and BNF), an unsigned numeric is a string (??).
    		 *       In such case, an error should be raised: */
    		try {
    			parser.parseQuery("SELECT LOWER(123) FROM foo");
    			fail("LOWER can not take a numeric in parameter.");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals("Incorrect argument: The ADQL function LOWER must have one parameter of type VARCHAR (i.e. a String)!", ex.getMessage());
    		}
    
    		// CASE: Idem for a second parameter:
    		try {
    			parser.parseQuery("SELECT IN_UNIT(12.3, 123) FROM foo");
    			fail("IN_UNIT can not take a numeric in 2nd parameter.");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals("Incorrect argument: The 2nd argument of the ADQL function IN_UNIT (i.e. target unit) must be of type VARCHAR (i.e. a string)!", ex.getMessage());
    		}
    	}
    
    	@Test
    	public void testOffset() {
    
    		// CASE: No OFFSET in ADQL-2.0
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_0);
    		try {
    			parser.parseQuery("SELECT * FROM foo ORDER BY id OFFSET 10");
    			fail("OFFSET should not be allowed with ADQL-2.0!");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals(" Encountered \"OFFSET\". Was expecting one of: <EOF> \",\" \";\" \"ASC\" \"DESC\" ", ex.getMessage());
    		}
    
    		// CASE: OFFSET allowed in ADQL-2.1
    		parser = new ADQLParser(ADQLVersion.V2_1);
    		try {
    			assertEquals("SELECT *\nFROM foo\nOFFSET 10", parser.parseQuery("SELECT * FROM foo OFFSET 10").toADQL());
    			assertEquals("SELECT *\nFROM foo\nORDER BY id ASC\nOFFSET 10", parser.parseQuery("SELECT * FROM foo ORDER BY id OFFSET 10").toADQL());
    			assertEquals("SELECT *\nFROM foo\nORDER BY id ASC\nOFFSET 0", parser.parseQuery("SELECT * FROM foo ORDER BY id OFFSET 0").toADQL());
    			assertEquals("SELECT TOP 5 *\nFROM foo\nORDER BY id ASC\nOFFSET 10", parser.parseQuery("SELECT TOP 5 * FROM foo ORDER BY id OFFSET 10").toADQL());
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error with a valid OFFSET syntax! (see console for more details)");
    		}
    
    		// CASE: Only an unsigned integer constant is allowed
    		String[] offsets = new String[]{ "-1", "colOffset", "2*5" };
    		String[] expectedErrors = new String[]{ " Encountered \"-\". Was expecting: <UNSIGNED_INTEGER> ", " Encountered \"colOffset\". Was expecting: <UNSIGNED_INTEGER> ", " Encountered \"*\". Was expecting one of: <EOF> \";\" " };
    		for(int i = 0; i < offsets.length; i++) {
    			try {
    				parser.parseQuery("SELECT * FROM foo OFFSET " + offsets[i]);
    				fail("Incorrect offset expression (\"" + offsets[i] + "\"). This test should have failed.");
    			} catch(Exception ex) {
    				assertEquals(ParseException.class, ex.getClass());
    				assertEquals(expectedErrors[i], ex.getMessage());
    			}
    		}
    	}
    
    	@Test
    	public void testGroupBy() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			// CASE: simple or qualified column name => OK!
    			try {
    				parser.parseQuery("SELECT * FROM cat GROUP BY oid;");
    				parser.parseQuery("SELECT * FROM cat GROUP BY cat.oid;");
    			} catch(Exception e) {
    				e.printStackTrace(System.err);
    				fail("These ADQL queries are strictly correct! No error should have occured. (see stdout for more details)");
    			}
    
    			/* CASE: in ADQL-2.0, only a column is allowed (on the contrary to
    			 *       ORDER BY which allows also an index of a selected column)
    			 *       => ERROR! */
    			final String Q_INDEX = "SELECT * FROM cat GROUP BY 1;";
    			if (version == ADQLVersion.V2_0) {
    				try {
    					// GROUP BY with a SELECT item index
    					parser.parseQuery(Q_INDEX);
    					fail("A SELECT item index is forbidden in GROUP BY! This test should have failed.");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \"1\". Was expecting one of: \"\\\"\" <REGULAR_IDENTIFIER_CANDIDATE> ", e.getMessage());
    				}
    			} else {
    				try {
    					parser.parseQuery(Q_INDEX);
    				} catch(Exception e) {
    					e.printStackTrace(System.err);
    					fail("These ADQL queries are strictly correct! No error should have occured. (see stdout for more details)");
    				}
    			}
    
    			// CASE: only from ADQL-2.1, any ADQL expression/operand => OK!
    			final String Q1 = "SELECT * FROM cat GROUP BY 1+2;", Q2 = "SELECT * FROM cat GROUP BY sqrt(foo);";
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery(Q1);
    					fail("In ADQL-v2.0, GROUP BY with an expression is forbidden!");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \"1\". Was expecting one of: \"\\\"\" <REGULAR_IDENTIFIER_CANDIDATE> ", e.getMessage());
    				}
    				try {
    					parser.parseQuery(Q2);
    					fail("In ADQL-v2.0, GROUP BY with an expression is forbidden!");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \"sqrt\". Was expecting one of: \"\\\"\" <REGULAR_IDENTIFIER_CANDIDATE> \n" + "(HINT: \"sqrt\" is a reserved ADQL word in v2.0. To use it as a column/table/schema name/alias, write it between double quotes.)", e.getMessage());
    				}
    			} else {
    				try {
    					parser.parseQuery(Q1);
    					parser.parseQuery(Q2);
    				} catch(Exception e) {
    					e.printStackTrace(System.err);
    					fail("In ADQL-" + version + ", GROUP BY with expression SHOULD be allowed. (see console for more details)");
    				}
    			}
    		}
    	}
    
    	@Test
    	public void testOrderBy() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    			try {
    				// CASE: Simple column name
    				ADQLQuery query;
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY oid;");
    				assertNotNull(query.getOrderBy().get(0).getExpression());
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY oid ASC;");
    				assertNotNull(query.getOrderBy().get(0).getExpression());
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY oid DESC;");
    				assertNotNull(query.getOrderBy().get(0).getExpression());
    
    				// CASE: selected column reference
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY 1;");
    				assertNotNull(query.getOrderBy().get(0).getColumnReference());
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY 1 ASC;");
    				assertNotNull(query.getOrderBy().get(0).getColumnReference());
    				query = parser.parseQuery("SELECT * FROM cat ORDER BY 1 DESC;");
    				assertNotNull(query.getOrderBy().get(0).getColumnReference());
    			} catch(Exception e) {
    				e.printStackTrace(System.err);
    				fail("These ADQL queries are strictly correct! No error should have occured. (cf console for more details)");
    			}
    
    			// CASE: only in ADQL-2.0, qualified columns are forbidden
    			String Q_QUAL_COL = "SELECT * FROM cat ORDER BY cat.oid;";
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery(Q_QUAL_COL);
    					fail("In ADQL-v2.0, a qualified column name is forbidden in ORDER BY! This test should have failed.");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \".\". Was expecting one of: <EOF> \",\" \";\" \"ASC\" \"DESC\" ", e.getMessage());
    				}
    			} else {
    				try {
    					parser.parseQuery(Q_QUAL_COL);
    				} catch(Exception e) {
    					e.printStackTrace();
    					fail("In ADQL-" + version + ", ORDER BY with a qualified column name should be allowed! (see console for more details)");
    				}
    			}
    
    			// CASE: Query reported as in error before a bug correction:
    			/*
    			 * NOTE: same as above but with a real bug report.
    			 */
    			Q_QUAL_COL = "SELECT TOP 10 browndwarfs.cat.jmag FROM browndwarfs.cat ORDER BY browndwarfs.cat.jmag";
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery(Q_QUAL_COL);
    					fail("A qualified column name is forbidden in ORDER BY! This test should have failed.");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \".\". Was expecting one of: <EOF> \",\" \";\" \"ASC\" \"DESC\" ", e.getMessage());
    				}
    			} else {
    				try {
    					parser.parseQuery(Q_QUAL_COL);
    				} catch(Exception e) {
    					e.printStackTrace();
    					fail("In ADQL-" + version + ", ORDER BY with a qualified column name should be allowed! (see console for more details)");
    				}
    			}
    
    			// CASE: only from ADQL-2.1, any ADQL expression/operand => OK!
    			final String Q1 = "SELECT * FROM cat ORDER BY 1+2;", Q2 = "SELECT * FROM cat ORDER BY sqrt(foo);";
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery(Q1);
    					fail("In ADQL-v2.0, ORDER BY with an expression is forbidden!");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \"+\". Was expecting one of: <EOF> \",\" \";\" \"ASC\" \"DESC\" ", e.getMessage());
    				}
    				try {
    					parser.parseQuery(Q2);
    					fail("In ADQL-v2.0, ORDER BY with an expression is forbidden!");
    				} catch(Exception e) {
    					assertEquals(ParseException.class, e.getClass());
    					assertEquals(" Encountered \"sqrt\". Was expecting one of: <UNSIGNED_INTEGER> \"\\\"\" <REGULAR_IDENTIFIER_CANDIDATE> \n" + "(HINT: \"sqrt\" is a reserved ADQL word in v2.0. To use it as a column/table/schema name/alias, write it between double quotes.)", e.getMessage());
    				}
    			} else {
    				try {
    					parser.parseQuery(Q1);
    					parser.parseQuery(Q2);
    				} catch(Exception e) {
    					e.printStackTrace(System.err);
    					fail("In ADQL-" + version + ", ORDER BY with expression SHOULD be allowed. (see console for more details)");
    				}
    			}
    		}
    	}
    
    	@Test
    	public void testJoinUsing() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    			try {
    				// JOIN ... USING(...)
    				parser.parseQuery("SELECT * FROM cat JOIN cat2 USING(oid);");
    			} catch(Exception e) {
    				e.printStackTrace(System.err);
    				fail("This ADQL query is strictly correct! No error should have occured. (see stdout for more details)");
    			}
    			try {
    				// JOIN ... USING(...)
    				parser.parseQuery("SELECT * FROM cat JOIN cat2 USING(cat.oid);");
    				fail("A qualified column name is forbidden in USING(...)! This test should have failed.");
    			} catch(Exception e) {
    				assertEquals(ParseException.class, e.getClass());
    				assertEquals(" Encountered \".\". Was expecting one of: \")\" \",\" ", e.getMessage());
    			}
    
    			try {
    				// JOIN ... USING(...)
    				parser.parseQuery("SELECT * FROM cat JOIN cat2 USING(1);");
    				fail("A column index is forbidden in USING(...)! This test should have failed.");
    			} catch(Exception e) {
    				assertEquals(ParseException.class, e.getClass());
    				assertEquals(" Encountered \"1\". Was expecting one of: \"\\\"\" <REGULAR_IDENTIFIER_CANDIDATE> ", e.getMessage());
    			}
    		}
    	}
    
    	@Test
    	public void testDelimitedIdentifiersWithDot() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    			try {
    				ADQLQuery query = parser.parseQuery("SELECT * FROM \"B/avo.rad/catalog\";");
    				assertEquals("B/avo.rad/catalog", query.getFrom().getTables().get(0).getTableName());
    			} catch(Exception e) {
    				e.printStackTrace(System.err);
    				fail("The ADQL query is strictly correct! No error should have occured. (see stdout for more details)");
    			}
    		}
    	}
    
    	@Test
    	public void testJoinTree() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    			try {
    				String[] queries = new String[]{ "SELECT * FROM aTable A JOIN aSecondTable B ON A.id = B.id JOIN aThirdTable C ON B.id = C.id;", "SELECT * FROM aTable A NATURAL JOIN aSecondTable B NATURAL JOIN aThirdTable C;" };
    				for(String q : queries) {
    					ADQLQuery query = parser.parseQuery(q);
    
    					assertTrue(query.getFrom() instanceof ADQLJoin);
    
    					ADQLJoin join = ((ADQLJoin)query.getFrom());
    					assertTrue(join.getLeftTable() instanceof ADQLJoin);
    					assertTrue(join.getRightTable() instanceof ADQLTable);
    					assertEquals("aThirdTable", ((ADQLTable)join.getRightTable()).getTableName());
    
    					join = (ADQLJoin)join.getLeftTable();
    					assertTrue(join.getLeftTable() instanceof ADQLTable);
    					assertEquals("aTable", ((ADQLTable)join.getLeftTable()).getTableName());
    					assertTrue(join.getRightTable() instanceof ADQLTable);
    					assertEquals("aSecondTable", ((ADQLTable)join.getRightTable()).getTableName());
    				}
    			} catch(Exception e) {
    				e.printStackTrace(System.err);
    				fail("The ADQL query is strictly correct! No error should have occured. (see stdout for more details)");
    			}
    		}
    	}
    
    	@Test
    	public void test() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    			try {
    				ADQLQuery query = parser.parseQuery("SELECT 'truc''machin'  	'bidule' --- why not a comment now ^^\n'FIN' FROM foo;");
    				assertNotNull(query);
    				assertEquals("truc'machinbiduleFIN", ((StringConstant)(query.getSelect().get(0).getOperand())).getValue());
    				assertEquals("'truc''machinbiduleFIN'", query.getSelect().get(0).getOperand().toADQL());
    			} catch(Exception ex) {
    				fail("String litteral concatenation is perfectly legal according to the ADQL standard.");
    			}
    
    			// With a comment ending the query
    			try {
    				ADQLQuery query = parser.parseQuery("SELECT TOP 1 * FROM ivoa.ObsCore -- comment");
    				assertNotNull(query);
    			} catch(Exception ex) {
    				ex.printStackTrace();
    				fail("String litteral concatenation is perfectly legal according to the ADQL standard.");
    			}
    		}
    	}
    
    	@Test
    	public void testIncorrectCharacter() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			/* An identifier must be written only with digits, an underscore or
    			 * regular latin characters: */
    			try {
    				(new ADQLParser(version)).parseQuery("select gr\u00e9gory FROM aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().startsWith("Incorrect character encountered at l.1, c.10: "));
    				assertTrue(t.getMessage().endsWith("Possible cause: a non-ASCI/UTF-8 character (solution: remove/replace it)."));
    			}
    
    			/* Un-finished double/single quoted string: */
    			try {
    				(new ADQLParser(version)).parseQuery("select \"stuff FROM aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().startsWith("Incorrect character encountered at l.1, c.26: <EOF>"));
    				assertTrue(t.getMessage().endsWith("Possible cause: a string between single or double quotes which is never closed (solution: well...just close it!)."));
    			}
    
    			// But in a string, delimited identifier or a comment, it is fine:
    			try {
    				(new ADQLParser(version)).parseQuery("select 'gr\u00e9gory' FROM aTable");
    				(new ADQLParser(version)).parseQuery("select \"gr\u00e9gory\" FROM aTable");
    				(new ADQLParser(version)).parseQuery("select * FROM aTable -- a comment by Gr\u00e9gory");
    			} catch(Throwable t) {
    				fail("This error should never occurs because all these queries have an accentuated character but at a correct place.");
    			}
    		}
    	}
    
    	@Test
    	public void testMultipleSpacesInOrderAndGroupBy() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			try {
    				ADQLParser parser = new ADQLParser(version);
    
    				// Single space:
    				parser.parseQuery("select * from aTable ORDER BY aCol");
    				parser.parseQuery("select * from aTable GROUP BY aCol");
    
    				// More than one space:
    				parser.parseQuery("select * from aTable ORDER      BY aCol");
    				parser.parseQuery("select * from aTable GROUP      BY aCol");
    
    				// With any other space character:
    				parser.parseQuery("select * from aTable ORDER\tBY aCol");
    				parser.parseQuery("select * from aTable ORDER\nBY aCol");
    				parser.parseQuery("select * from aTable ORDER \t\nBY aCol");
    
    				parser.parseQuery("select * from aTable GROUP\tBY aCol");
    				parser.parseQuery("select * from aTable GROUP\nBY aCol");
    				parser.parseQuery("select * from aTable GROUP \t\nBY aCol");
    			} catch(Throwable t) {
    				t.printStackTrace();
    				fail("Having multiple space characters between the ORDER/GROUP and the BY keywords should not generate any parsing error.");
    			}
    		}
    	}
    
    	@Test
    	public void testADQLReservedWord() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			final String hintAbs = ".*\n\\(HINT: \"abs\" is a reserved ADQL word in v[0-9]+\\.[0-9]+\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)";
    			final String hintPoint = ".*\n\\(HINT: \"point\" is a reserved ADQL word in v[0-9]+\\.[0-9]+\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)";
    			final String hintExists = ".*\n\\(HINT: \"exists\" is a reserved ADQL word in v[0-9]+\\.[0-9]+\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)";
    			final String hintLike = ".*\n\\(HINT: \"LIKE\" is a reserved ADQL word in v[0-9]+\\.[0-9]+\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)";
    
    			/* TEST AS A COLUMN/TABLE/SCHEMA NAME... */
    			// ...with a numeric function name (but no param):
    			try {
    				parser.parseQuery("select abs from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintAbs));
    			}
    			// ...with a geometric function name (but no param):
    			try {
    				parser.parseQuery("select point from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintPoint));
    			}
    			// ...with an ADQL function name (but no param):
    			try {
    				parser.parseQuery("select exists from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintExists));
    			}
    			// ...with an ADQL syntax item:
    			try {
    				parser.parseQuery("select LIKE from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintLike));
    			}
    
    			/* TEST AS AN ALIAS... */
    			// ...with a numeric function name (but no param):
    			try {
    				parser.parseQuery("select aCol AS abs from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintAbs));
    			}
    			// ...with a geometric function name (but no param):
    			try {
    				parser.parseQuery("select aCol AS point from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintPoint));
    			}
    			// ...with an ADQL function name (but no param):
    			try {
    				parser.parseQuery("select aCol AS exists from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintExists));
    			}
    			// ...with an ADQL syntax item:
    			try {
    				parser.parseQuery("select aCol AS LIKE from aTable");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintLike));
    			}
    
    			/* TEST AT THE END OF THE QUERY (AND IN A WHERE) */
    			try {
    				parser.parseQuery("select aCol from aTable WHERE toto = abs");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(hintAbs));
    			}
    		}
    	}
    
    	@Test
    	public void testSQLReservedWord() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			try {
    				parser.parseQuery("SELECT rows FROM aTable");
    				fail("\"ROWS\" is an SQL reserved word. This query should not pass.");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(".*\n\\(HINT: \"rows\" is not supported in ADQL v[0-9]+\\.[0-9]+, but is however a reserved word\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)"));
    			}
    
    			try {
    				parser.parseQuery("SELECT CASE WHEN aCol = 2 THEN 'two' ELSE 'smth else' END as str FROM aTable");
    				fail("ADQL does not support the CASE syntax. This query should not pass.");
    			} catch(Throwable t) {
    				assertEquals(ParseException.class, t.getClass());
    				assertTrue(t.getMessage().matches(".*\n\\(HINT: \"CASE\" is not supported in ADQL v[0-9]+\\.[0-9]+, but is however a reserved word\\. To use it as a column/table/schema name/alias, write it between double quotes\\.\\)"));
    			}
    		}
    	}
    
    	@Test
    	public void testUDFName() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			// CASE: Valid UDF name => OK
    			try {
    				parser.parseQuery("SELECT foo(p1,p2) FROM aTable");
    			} catch(Throwable t) {
    				t.printStackTrace();
    				fail("Unexpected parsing error! This query should have passed. (see console for more details)");
    			}
    
    			// CASE: Invalid UDF name => ParseException
    			final String[] functionsToTest = new String[]{ "_foo", "2do", "do?" };
    			for(String fct : functionsToTest) {
    				try {
    					parser.parseQuery("SELECT " + fct + "(p1,p2) FROM aTable");
    					fail("A UDF name like \"" + fct + "\" is not allowed by the ADQL grammar. This query should not pass.");
    				} catch(Throwable t) {
    					assertEquals(ParseException.class, t.getClass());
    					assertEquals("Invalid (User Defined) Function name: \"" + fct + "\"!", t.getMessage());
    				}
    			}
    		}
    	}
    
    	@Test
    	public void testUDFDeclaration() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			ADQLParser parser = new ADQLParser(version);
    
    			// CASE: Any UDF allowed => OK!
    			parser.getSupportedFeatures().allowAnyUdf(true);
    			try {
    				assertNotNull(parser.parseQuery("SELECT foo(1,2) FROM bar"));
    			} catch(Throwable t) {
    				t.printStackTrace();
    				fail("Unexpected parsing error! This query should have passed. (see console for more details)");
    			}
    
    			// CASE: No UDF allowed => ERROR
    			parser.getSupportedFeatures().allowAnyUdf(false);
    			try {
    				parser.parseQuery("SELECT foo(1,2) FROM bar");
    				fail("No UDF is allowed. This query should have failed!");
    			} catch(Throwable t) {
    				assertEquals(UnresolvedIdentifiersException.class, t.getClass());
    				assertEquals("1 unsupported expressions!\n  - Unsupported ADQL feature: \"foo(param1 ?type?, param2 ?type?) -> ?type?\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-udf')!", t.getMessage());
    			}
    
    			// CASE: a single UDF declared => OK!
    			try {
    				parser.getSupportedFeatures().support(FunctionDef.parse("foo(i1 INTEGER, i2 INTEGER) -> INTEGER").toLanguageFeature());
    				assertNotNull(parser.parseQuery("SELECT foo(1,2) FROM bar"));
    			} catch(Throwable t) {
    				t.printStackTrace();
    				fail("Unexpected parsing error! This query should have passed. (see console for more details)");
    			}
    		}
    	}
    
    	@Test
    	public void testOptionalFeatures() {
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_0);
    
    		// CASE: No support for the ADQL-2.1 function - LOWER => ERROR
    		try {
    			parser.parseQuery("SELECT LOWER(foo) FROM aTable");
    			fail("The function \"LOWER\" is not yet supported in ADQL-2.0. This query should not pass.");
    		} catch(Throwable t) {
    			assertEquals(ParseException.class, t.getClass());
    			assertTrue(t.getMessage().contains("(HINT: \"LOWER\" is not supported in ADQL v2.0, but is however a reserved word."));
    		}
    
    		// CASE: LOWER supported by default in ADQL-2.1 => OK
    		parser = new ADQLParser(ADQLVersion.V2_1);
    		try {
    			ADQLQuery q = parser.parseQuery("SELECT LOWER(foo) FROM aTable");
    			assertNotNull(q);
    			assertEquals("SELECT LOWER(foo)\nFROM aTable", q.toADQL());
    		} catch(Throwable t) {
    			t.printStackTrace();
    			fail("The function \"LOWER\" is available in ADQL-2.1 and is declared as supported. This query should pass.");
    		}
    
    		// CASE: LOWER now declared as not supported => ERROR
    		assertTrue(parser.getSupportedFeatures().unsupport(LowerFunction.FEATURE));
    		try {
    			parser.parseQuery("SELECT LOWER(foo) FROM aTable");
    			fail("The function \"LOWER\" is not available in ADQL-2.1 and is here declared as not supported. This query should not pass.");
    		} catch(Throwable t) {
    			assertEquals(UnresolvedIdentifiersException.class, t.getClass());
    			UnresolvedIdentifiersException uie = (UnresolvedIdentifiersException)t;
    			assertEquals(1, uie.getNbErrors());
    			Exception err = uie.getErrors().next();
    			assertEquals(UnsupportedFeatureException.class, err.getClass());
    			assertEquals("Unsupported ADQL feature: \"LOWER\" (of type '" + LanguageFeature.TYPE_ADQL_STRING + "')!", err.getMessage());
    		}
    
    		/* ***************************************************************** */
    		/* NOTE: Geometrical functions are the only optional features in 2.0 */
    		/* ***************************************************************** */
    
    		parser = new ADQLParser(ADQLVersion.V2_0);
    
    		// CASE: By default all geometries are supported so if one is used => OK
    		try {
    			assertNotNull(parser.parseQuery("SELECT POINT('', ra, dec) FROM aTable"));
    		} catch(Throwable t) {
    			t.printStackTrace();
    			fail("Unexpected error! This query should have passed. (see console for more details)");
    		}
    
    		// unsupport all features:
    		parser.getSupportedFeatures().unsupportAll();
    
    		// CASE: No geometry supported so if one is used => ERROR
    		try {
    			parser.parseQuery("SELECT POINT('', ra, dec) FROM aTable");
    			fail("The geometrical function \"POINT\" is not declared. This query should not pass.");
    		} catch(Throwable t) {
    			assertEquals(UnresolvedIdentifiersException.class, t.getClass());
    			UnresolvedIdentifiersException allErrors = (UnresolvedIdentifiersException)t;
    			assertEquals(1, allErrors.getNbErrors());
    			assertEquals("Unsupported ADQL feature: \"POINT\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", allErrors.getErrors().next().getMessage());
    		}
    
    		// now support only POINT:
    		assertTrue(parser.getSupportedFeatures().support(PointFunction.FEATURE));
    
    		// CASE: Just supporting the only used geometry => OK
    		try {
    			assertNotNull(parser.parseQuery("SELECT POINT('', ra, dec) FROM aTable"));
    		} catch(Throwable t) {
    			t.printStackTrace();
    			fail("Unexpected error! This query should have passed. (see console for more details)");
    		}
    	}
    
    	@Test
    	public void testGeometry() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			// DECLARE A SIMPLE PARSER where all geometries are allowed by default:
    			ADQLParser parser = new ADQLParser(version);
    
    			// Test with several geometries while all are allowed:
    			try {
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("This query contains several geometries, and all are theoretically allowed: this test should have succeeded!");
    			}
    
    			// Test with several geometries while only the allowed ones:
    			try {
    				parser = new ADQLParser(version);
    				parser.getSupportedFeatures().unsupportAll(LanguageFeature.TYPE_ADQL_GEO);
    				parser.getSupportedFeatures().support(ContainsFunction.FEATURE);
    				parser.getSupportedFeatures().support(PointFunction.FEATURE);
    				parser.getSupportedFeatures().support(CircleFunction.FEATURE);
    
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("This query contains several geometries, and all are theoretically allowed: this test should have succeeded!");
    			}
    			try {
    				parser.parseQuery("SELECT * FROM foo WHERE INTERSECTS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;");
    				fail("This query contains a not-allowed geometry function (INTERSECTS): this test should have failed!");
    			} catch(ParseException pe) {
    				assertTrue(pe instanceof UnresolvedIdentifiersException);
    				UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    				assertEquals(1, ex.getNbErrors());
    				assertEquals("Unsupported ADQL feature: \"INTERSECTS\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", ex.getErrors().next().getMessage());
    			}
    
    			// TODO Test by adding REGION: // Only possible with ADQL-2.0 since in ADQL-2.1, REGION has been removed!
    			try {
    				parser = new ADQLParser(ADQLVersion.V2_0);
    				parser.getSupportedFeatures().unsupportAll(LanguageFeature.TYPE_ADQL_GEO);
    				parser.getSupportedFeatures().support(ContainsFunction.FEATURE);
    				parser.getSupportedFeatures().support(PointFunction.FEATURE);
    				parser.getSupportedFeatures().support(CircleFunction.FEATURE);
    				parser.getSupportedFeatures().support(RegionFunction.FEATURE);
    
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(REGION('Position 12.3 45.6'), REGION('circle 1.2 2.3 5')) = 1;"));
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("[ADQL-" + parser.getADQLVersion() + "] This query contains several geometries, and all are theoretically allowed: this test should have succeeded!");
    			}
    			try {
    				parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(REGION('Position 12.3 45.6'), REGION('BOX 1.2 2.3 5 9')) = 1;");
    				fail("This query contains a not-allowed geometry function (BOX): this test should have failed!");
    			} catch(ParseException pe) {
    				assertTrue(pe instanceof UnresolvedIdentifiersException);
    				UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    				assertEquals(1, ex.getNbErrors());
    				assertEquals("Unsupported STC-s region type: \"BOX\" (equivalent to the ADQL feature \"BOX\" of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", ex.getErrors().next().getMessage());
    			}
    
    			// Test with several geometries while none geometry is allowed:
    			try {
    				parser = new ADQLParser(version);
    				parser.getSupportedFeatures().unsupportAll(LanguageFeature.TYPE_ADQL_GEO);
    
    				parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;");
    				fail("This query contains geometries while they are all forbidden: this test should have failed!");
    			} catch(ParseException pe) {
    				assertTrue(pe instanceof UnresolvedIdentifiersException);
    				UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    				assertEquals(3, ex.getNbErrors());
    				Iterator<ParseException> itErrors = ex.getErrors();
    				assertEquals("Unsupported ADQL feature: \"CONTAINS\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", itErrors.next().getMessage());
    				assertEquals("Unsupported ADQL feature: \"POINT\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", itErrors.next().getMessage());
    				assertEquals("Unsupported ADQL feature: \"CIRCLE\" (of type 'ivo://ivoa.net/std/TAPRegExt#features-adql-geo')!", itErrors.next().getMessage());
    			}
    		}
    	}
    
    	@Test
    	public void testGeometryWithOptionalArgs() {
    		/*
    		 * NOTE:
    		 * 	Since ADQL-2.1, the coordinate system argument becomes optional.
    		 *  Besides, BOX, CIRCLE and POLYGON can now accept POINTs instead of
    		 *  pairs of coordinates.
    		 */
    
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_1);
    
    		// CASE: with no coordinate system => equivalent to coosys = ''
    		try {
    			assertEquals("POINT('', 1, 2)", parser.parseSelect("SELECT POINT(1, 2)").get(0).toADQL());
    
    			assertEquals("CIRCLE('', 1, 2, 3)", parser.parseSelect("SELECT CIRCLE(1, 2, 3)").get(0).toADQL());
    			assertEquals("CIRCLE('', POINT('', 1, 2), 3)", parser.parseSelect("SELECT CIRCLE(POINT(1,2), 3)").get(0).toADQL());
    			assertEquals("CIRCLE('', colCenter, 3)", parser.parseSelect("SELECT CIRCLE(colCenter, 3)").get(0).toADQL());
    
    			assertEquals("BOX('', 1, 2, 3, 4)", parser.parseSelect("SELECT BOX(1, 2, 3, 4)").get(0).toADQL());
    			assertEquals("BOX('', POINT('', 1, 2), 3, 4)", parser.parseSelect("SELECT BOX(POINT(1, 2), 3, 4)").get(0).toADQL());
    			assertEquals("BOX('', colCenter, 3, 4)", parser.parseSelect("SELECT BOX(colCenter, 3, 4)").get(0).toADQL());
    
    			assertEquals("POLYGON('', 1, 2, 3, 4, 5, 6)", parser.parseSelect("SELECT POLYGON(1, 2, 3, 4, 5, 6)").get(0).toADQL());
    			assertEquals("POLYGON('', POINT('', 1, 2), POINT('', 3, 4), POINT('', 5, 6))", parser.parseSelect("SELECT POLYGON(POINT(1, 2), POINT(3, 4), POINT(5, 6))").get(0).toADQL());
    			assertEquals("POLYGON('', point1, point2, point3)", parser.parseSelect("SELECT POLYGON(point1, point2, point3)").get(0).toADQL());
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error! All parsed geometries are correct.");
    		}
    
    		// CASE: wrong nb of arguments for POLYGON
    		for(String wrongQuery : new String[]{ "SELECT POLYGON(ra, dec, 3, 4, 5)", "SELECT POLYGON(ra, dec, 3, 4, 5, 6, 7)", "SELECT POLYGON(p1, p2)" })
    			try {
    				parser.parseSelect(wrongQuery);
    				fail("Impossible to create a POLYGON with an incomplete list of vertices! The last point is missing or incomplete.");
    			} catch(Exception ex) {
    				assertEquals(ParseException.class, ex.getClass());
    				assertTrue(ex.getMessage().trim().startsWith("Encountered \")\"."));
    			}
    	}
    
    	@Test
    	public void testCoordSys() {
    		for(ADQLVersion version : ADQLVersion.values()) {
    			// DECLARE A SIMPLE PARSER where all coordinate systems are allowed by default:
    			ADQLParser parser = new ADQLParser(version);
    
    			// A coordinate system MUST be a string literal:
    			try {
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('From ' || 'here', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    				fail("A coordinate system can NOT be a string concatenation!");
    			} catch(ParseException pe) {
    				assertEquals(ParseException.class, pe.getClass());
    				assertEquals(48, pe.getPosition().beginColumn);
    			}
    			try {
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT(aColumn, 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    				fail("A coordinate system can NOT be a column reference!");
    			} catch(ParseException pe) {
    				assertEquals(ParseException.class, pe.getClass());
    				assertEquals((version == ADQLVersion.V2_0 ? 40 : 53), pe.getPosition().beginColumn);
    			}
    
    			// Test with several coordinate systems while all are allowed:
    			try {
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('icrs', 12.3, 45.6), CIRCLE('cartesian2', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('lsr', 12.3, 45.6), CIRCLE('galactic heliocenter', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('unknownframe', 12.3, 45.6), CIRCLE('galactic unknownrefpos spherical2', 1.2, 2.3, 5)) = 1;"));
    				if (version == ADQLVersion.V2_0) {
    					assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(REGION('position icrs lsr 12.3 45.6'), REGION('circle fk5 1.2 2.3 5')) = 1;"));
    					assertNotNull(parser.parseQuery("SELECT Region('not(position 1 2)') FROM foo;"));
    				}
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("This query contains several valid coordinate systems, and all are theoretically allowed: this test should have succeeded!");
    			}
    
    			// Test with several coordinate systems while only some allowed:
    			try {
    				parser = new ADQLParser(version);
    				parser.setAllowedCoordSys(Arrays.asList(new String[]{ "icrs * *", "fk4 geocenter *", "galactic * spherical2" }));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('icrs', 12.3, 45.6), CIRCLE('cartesian3', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT POINT('fk4', 12.3, 45.6) FROM foo;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('fk4 geocenter', 12.3, 45.6), CIRCLE('cartesian2', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('galactic', 12.3, 45.6), CIRCLE('galactic spherical2', 1.2, 2.3, 5)) = 1;"));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('galactic geocenter', 12.3, 45.6), CIRCLE('galactic lsr spherical2', 1.2, 2.3, 5)) = 1;"));
    				if (version == ADQLVersion.V2_0) {
    					assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(REGION('position galactic lsr 12.3 45.6'), REGION('circle icrs 1.2 2.3 5')) = 1;"));
    					assertNotNull(parser.parseQuery("SELECT Region('not(position 1 2)') FROM foo;"));
    				}
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("This query contains several valid coordinate systems, and all are theoretically allowed: this test should have succeeded!");
    			}
    			try {
    				parser.parseQuery("SELECT POINT('fk5 geocenter', 12.3, 45.6) FROM foo;");
    				fail("This query contains a not-allowed coordinate system ('fk5' is not allowed): this test should have failed!");
    			} catch(ParseException pe) {
    				assertTrue(pe instanceof UnresolvedIdentifiersException);
    				UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    				assertEquals(1, ex.getNbErrors());
    				assertEquals("Coordinate system \"fk5 geocenter\" (= \"FK5 GEOCENTER SPHERICAL2\") not allowed in this implementation. Allowed coordinate systems are: fk4 geocenter *, galactic * spherical2, icrs * *", ex.getErrors().next().getMessage());
    			}
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery("SELECT Region('not(position fk5 heliocenter 1 2)') FROM foo;");
    					fail("This query contains a not-allowed coordinate system ('fk5' is not allowed): this test should have failed!");
    				} catch(ParseException pe) {
    					assertTrue(pe instanceof UnresolvedIdentifiersException);
    					UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    					assertEquals(1, ex.getNbErrors());
    					assertEquals("Coordinate system \"FK5 HELIOCENTER\" (= \"FK5 HELIOCENTER SPHERICAL2\") not allowed in this implementation. Allowed coordinate systems are: fk4 geocenter *, galactic * spherical2, icrs * *", ex.getErrors().next().getMessage());
    				}
    			}
    
    			// Test with a coordinate system while none is allowed:
    			try {
    				parser = new ADQLParser(version);
    				parser.setAllowedCoordSys(new ArrayList<String>(0));
    				assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('', 12.3, 45.6), CIRCLE('', 1.2, 2.3, 5)) = 1;"));
    				if (version == ADQLVersion.V2_0) {
    					assertNotNull(parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(REGION('position 12.3 45.6'), REGION('circle 1.2 2.3 5')) = 1;"));
    					assertNotNull(parser.parseQuery("SELECT Region('not(position 1 2)') FROM foo;"));
    				}
    			} catch(ParseException pe) {
    				pe.printStackTrace();
    				fail("This query specifies none coordinate system: this test should have succeeded!");
    			}
    			try {
    				parser.parseQuery("SELECT * FROM foo WHERE CONTAINS(POINT('ICRS SPHERICAL2', 12.3, 45.6), CIRCLE('icrs', 1.2, 2.3, 5)) = 1;");
    				fail("This query specifies coordinate systems while they are all forbidden: this test should have failed!");
    			} catch(ParseException pe) {
    				assertTrue(pe instanceof UnresolvedIdentifiersException);
    				UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    				assertEquals(2, ex.getNbErrors());
    				Iterator<ParseException> itErrors = ex.getErrors();
    				assertEquals("Coordinate system \"ICRS SPHERICAL2\" (= \"ICRS UNKNOWNREFPOS SPHERICAL2\") not allowed in this implementation. No coordinate system is allowed!", itErrors.next().getMessage());
    				assertEquals("Coordinate system \"icrs\" (= \"ICRS UNKNOWNREFPOS SPHERICAL2\") not allowed in this implementation. No coordinate system is allowed!", itErrors.next().getMessage());
    			}
    			if (version == ADQLVersion.V2_0) {
    				try {
    					parser.parseQuery("SELECT Region('not(position fk4 1 2)') FROM foo;");
    					fail("This query specifies coordinate systems while they are all forbidden: this test should have failed!");
    				} catch(ParseException pe) {
    					assertTrue(pe instanceof UnresolvedIdentifiersException);
    					UnresolvedIdentifiersException ex = (UnresolvedIdentifiersException)pe;
    					assertEquals(1, ex.getNbErrors());
    					assertEquals("Coordinate system \"FK4\" (= \"FK4 UNKNOWNREFPOS SPHERICAL2\") not allowed in this implementation. No coordinate system is allowed!", ex.getErrors().next().getMessage());
    				}
    			}
    		}
    	}
    
    	@Test
    	public void testTokenize() {
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_0);
    
    		final String[] EMPTY_STRINGS = new String[]{ null, "", "  ", " 	 " };
    
    		// TEST: NULL or empty string with end at EOF => only one token=EOF
    		try {
    			for(String str : EMPTY_STRINGS) {
    				Token[] tokens = parser.tokenize(str, false);
    				assertEquals(1, tokens.length);
    				assertEquals(ADQLGrammar200Constants.EOF, tokens[0].kind);
    			}
    		} catch(Exception e) {
    			e.printStackTrace();
    			fail("Unexpected error when providing a NULL or empty string to tokenize! (see console for more details)");
    		}
    
    		// TEST: NULL or empty string with truncation at EOQ/EOF => empty array
    		try {
    			for(String str : EMPTY_STRINGS)
    				assertEquals(0, parser.tokenize(str, true).length);
    		} catch(Exception e) {
    			e.printStackTrace();
    			fail("Unexpected error when providing a NULL or empty string to tokenize! (see console for more details)");
    		}
    
    		// TEST: unknown token => ParseException
    		try {
    			parser.tokenize("grégory", false);
    			fail("No known token is provided. A ParseException was expected.");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals("Incorrect character encountered at l.1, c.3: \"\\u00e9\" ('é'), after : \"\"!" + System.getProperty("line.separator", "\n") + "Possible cause: a non-ASCI/UTF-8 character (solution: remove/replace it).", ex.getMessage());
    		}
    
    		// TEST: correct list of token => ok
    		final String TEST_STR = "SELECT FROM Where foo; join";
    		try {
    			Token[] tokens = parser.tokenize(TEST_STR, false);
    			assertEquals(7, tokens.length);
    			int[] expected = new int[]{ ADQLGrammar200Constants.SELECT, ADQLGrammar200Constants.FROM, ADQLGrammar200Constants.WHERE, ADQLGrammar200Constants.REGULAR_IDENTIFIER_CANDIDATE, ADQLGrammar200Constants.EOQ, ADQLGrammar200Constants.JOIN, ADQLGrammar200Constants.EOF };
    			for(int i = 0; i < tokens.length; i++)
    				assertEquals(expected[i], tokens[i].kind);
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error! All ADQL expressions were composed of correct tokens. (see console for more details)");
    		}
    
    		// TEST: same with truncation at EOQ/EOF => same but truncated from EOQ
    		try {
    			Token[] tokens = parser.tokenize(TEST_STR, true);
    			assertEquals(4, tokens.length);
    			int[] expected = new int[]{ ADQLGrammar200Constants.SELECT, ADQLGrammar200Constants.FROM, ADQLGrammar200Constants.WHERE, ADQLGrammar200Constants.REGULAR_IDENTIFIER_CANDIDATE };
    			for(int i = 0; i < tokens.length; i++)
    				assertEquals(expected[i], tokens[i].kind);
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error! All ADQL expressions were composed of correct tokens. (see console for more details)");
    		}
    	}
    
    	@Test
    	public void testDistance() {
    		// CASE: In ADQL-2.0, DISTANCE(POINT, POINT) is allowed:
    		ADQLParser parser = new ADQLParser(ADQLVersion.V2_0);
    		try {
    			assertEquals("DISTANCE(POINT('', ra, dec), POINT('', ra2, dec2))", parser.parseSelect("SELECT DISTANCE(POINT('', ra, dec), POINT('', ra2, dec2))").get(0).toADQL());
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error! All ADQL expressions were composed of correct tokens. (see console for more details)");
    		}
    
    		// CASE: ...BUT not DISTANCE(lon1, lat1, lon2, lat2)
    		try {
    			parser.parseSelect("SELECT DISTANCE(ra, dec, ra2, dec2)");
    			fail("In ADQL-2.0, DISTANCE(lon1, lat1, lon2, lat2) should not be allowed!");
    		} catch(Exception ex) {
    			assertEquals(ParseException.class, ex.getClass());
    			assertEquals(" Encountered \",\". Was expecting one of: \")\" \".\" \".\" \")\" ", ex.getMessage());
    		}
    
    		/* CASE: In ADQL-2.1 (and more), DISTANCE(POINT, POINT) and
    		 *       DISTANCE(lon1, lat1, lon2, lat2) are both allowed: */
    		parser = new ADQLParser(ADQLVersion.V2_1);
    		try {
    			assertEquals("DISTANCE(POINT('', ra, dec), POINT('', ra2, dec2))", parser.parseSelect("SELECT DISTANCE(POINT('', ra, dec), POINT('', ra2, dec2))").get(0).toADQL());
    			assertEquals("DISTANCE(POINT('', ra, dec), POINT('', ra2, dec2))", parser.parseSelect("SELECT DISTANCE(ra, dec, ra2, dec2)").get(0).toADQL());
    		} catch(Exception ex) {
    			ex.printStackTrace();
    			fail("Unexpected error! All ADQL expressions were composed of correct tokens. (see console for more details)");
    		}
    	}
    
    }