/* // $Id: ConnectionTest.java 550 2012-10-03 00:59:15Z jhyde $ // // Licensed to Julian Hyde under one or more contributor license // agreements. See the NOTICE file distributed with this work for // additional information regarding copyright ownership. // // Julian Hyde licenses this file to you under the Apache License, // Version 2.0 (the "License"); you may not use this file except in // compliance with the License. You may obtain a copy of the License at: // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. */ package org.olap4j; import org.olap4j.driver.xmla.XmlaOlap4jDriver; import org.olap4j.impl.Bug; import org.olap4j.impl.Olap4jUtil; import org.olap4j.mdx.*; import org.olap4j.mdx.parser.*; import org.olap4j.metadata.*; import org.olap4j.test.TestContext; import org.olap4j.test.TestContext.Tester; import org.olap4j.test.TestContext.Tester.Flavor; import org.olap4j.type.*; import junit.framework.AssertionFailedError; import junit.framework.TestCase; import java.io.*; import java.lang.reflect.InvocationTargetException; import java.sql.*; import java.util.*; import static org.olap4j.test.TestContext.nameList; /** * Unit test for olap4ld Driver and Connection classes. * * @author bkaempgen * @version $Id: ConnectionTest.java 550 2012-10-03 00:59:15Z jhyde $ */ public class ConnectionTest extends TestCase { private TestContext testContext = TestContext.instance(); private TestContext.Tester tester = testContext.getTester(); private static final boolean IS_JDK_16 = System.getProperty("java.version").startsWith("1.6."); /** * Simple strategy to prevent connection leaks: each test that needs a * connection assigns it to this field, and {@link #tearDown()} closes it * if it is not already closed. */ private Connection connection; protected void setUp() throws Exception { super.setUp(); } protected void tearDown() throws Exception { // Simple strategy to prevent connection leaks if (connection != null && !connection.isClosed()) { connection.close(); } connection = null; testContext = null; tester = null; } /** * Driver basics. */ public void testDriver() throws ClassNotFoundException, SQLException { Class<?> clazz = Class.forName(tester.getDriverClassName()); assertNotNull(clazz); assertTrue(Driver.class.isAssignableFrom(clazz)); // driver should have automatically registered itself Driver driver = DriverManager.getDriver(tester.getDriverUrlPrefix()); assertNotNull(driver); // deregister driver DriverManager.deregisterDriver(driver); try { Driver driver2 = DriverManager.getDriver(tester.getDriverUrlPrefix()); fail("expected error, got " + driver2); } catch (SQLException e) { assertEquals("No suitable driver", e.getMessage()); } // register explicitly DriverManager.registerDriver(driver); Driver driver3 = DriverManager.getDriver(tester.getDriverUrlPrefix()); assertNotNull(driver3); // test properties int majorVersion = driver.getMajorVersion(); int minorVersion = driver.getMinorVersion(); assertTrue(majorVersion >= 0); assertTrue(minorVersion >= 0); assertTrue(majorVersion > 0 || minorVersion > 0); // check that the getPropertyInfo method returns something sensible. // We can't test individual properties in this non-driver-specific test. DriverPropertyInfo[] driverPropertyInfos = driver.getPropertyInfo( tester.getDriverUrlPrefix(), new Properties()); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: break; default: assertTrue(driverPropertyInfos.length > 0); } } void assertIsValid(Connection connection, int timeout) { if (!IS_JDK_16) { return; } // We would like to evaluate // assertTrue(connection.isValid(0)); // but this code would not compile on JDK 1.5 or lower. So, we invoke // the same code by reflection. try { java.lang.reflect.Method method = Connection.class.getMethod("isValid", int.class); Boolean b = (Boolean) method.invoke(connection, timeout); assertTrue(b); } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (InvocationTargetException e) { if (e.getTargetException() instanceof AbstractMethodError) { // This happens in commons-dbcp. Somehow the method exists in // the connection class, but it fails later. Not the fault of // olap4j or the olapj driver, so ignore the error. Olap4jUtil.discard(e); } else { throw new RuntimeException(e); } } catch (AbstractMethodError e) { // This happens in commons-dbcp. Somehow the method exists in // the connection class, but it fails later. Not the fault of // olap4j or the olapj driver, so ignore the error. Olap4jUtil.discard(e); } } /** * Checks that the <code>isClosed</code> method of a Statement, ResultSet or * Connection object returns the expected result. Uses reflection because * the <code>isClosed</code> method only exists from JDBC 4.0 (JDK 1.6) * onwrds. * * @param o Connection, Statement or ResultSet object * @param b Expected result */ void assertIsClosed(Object o, boolean b) { if (!IS_JDK_16) { return; } if (tester.getWrapper() == TestContext.Wrapper.DBCP) { // commons-dbcp 1.1 doesn't support isClosed return; } // assertTrue(statment.isClosed()); try { Class<?> clazz; if (o instanceof Statement) { clazz = Statement.class; } else if (o instanceof ResultSet) { clazz = ResultSet.class; } else if (o instanceof Connection) { clazz = Connection.class; } else { throw new AssertionFailedError( "not a statement, resultSet or connection"); } java.lang.reflect.Method method = clazz.getMethod("isClosed"); Boolean closed = (Boolean) method.invoke(o); if (b) { assertTrue(closed); } else { assertFalse(closed); } } catch (NoSuchMethodException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } catch (InvocationTargetException e) { throw new RuntimeException(e); } } public void testConnection() throws ClassNotFoundException, SQLException { Class.forName(tester.getDriverClassName()); // connect using properties and no username/password connection = tester.createConnection(); assertNotNull(connection); // check isClosed, isValid assertFalse(connection.isClosed()); // check valid with no time limit assertIsValid(connection, 0); // check valid with one minute time limit; should be enough assertIsValid(connection, 60); connection.close(); assertTrue(connection.isClosed()); // it's ok to close twice switch (tester.getWrapper()) { case DBCP: // DBCP complains if you close a connection twice. Even though the // JDBC spec is clear that it is OK. break; default: connection.close(); break; } switch (tester.getFlavor()) { case MONDRIAN: // connect using username/password connection = tester.createConnectionWithUserPassword(); assertNotNull(connection); connection.close(); assertTrue(connection.isClosed()); // connect with URL only connection = DriverManager.getConnection(tester.getURL()); assertNotNull(connection); connection.close(); break; case XMLA: // in-process XMLA test does not support username/password break; case REMOTE_XMLA: // connect using username/password connection = tester.createConnectionWithUserPassword(); assertNotNull(connection); // connect with URL only connection = DriverManager.getConnection(tester.getURL()); assertNotNull(connection); connection.close(); break; } assertTrue(connection.isClosed()); } public void testConnectionUnwrap() throws SQLException { // commons-dbcp 1.1 doesn't do wrapping very well switch (tester.getWrapper()) { case DBCP: return; } connection = tester.createConnection(); // Trivial unwrap assertTrue(((OlapWrapper) connection).isWrapperFor(Connection.class)); Connection connection2 = ((OlapWrapper) connection).unwrap(Connection.class); assertEquals(connection2, connection); // Silly unwrap assertTrue(((OlapWrapper) connection).isWrapperFor(Object.class)); Object object = ((OlapWrapper) connection).unwrap(Object.class); assertEquals(object, connection); // Invalid unwrap assertFalse(((OlapWrapper) connection).isWrapperFor(Writer.class)); try { Writer writer = ((OlapWrapper) connection).unwrap(Writer.class); fail("expected exception, got writer" + writer); } catch (SQLException e) { assertTrue(e.getMessage().contains("does not implement")); } // Unwrap and get locale OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final Locale locale = olapConnection.getLocale(); assertEquals(locale, Locale.getDefault()); // Set locale to something else. olapConnection.setLocale(Locale.CANADA_FRENCH); assertEquals(olapConnection.getLocale(), Locale.CANADA_FRENCH); // Against mondrian, Sales cube is localized. final OlapDatabaseMetaData metaData = olapConnection.getMetaData(); final String databaseName = metaData.getDatabaseProductName(); final String databaseVersion = metaData.getDatabaseProductVersion(); if (databaseName.equals("Mondrian XML for Analysis Provider") && databaseVersion.compareTo("3.3") > 0) { olapConnection.setLocale(Locale.US); final Cube salesCubeUs = olapConnection.getOlapSchema().getCubes().get("Sales"); assertEquals("Sales", salesCubeUs.getCaption()); // Switch locales. Note that we have to re-read metadata from the // root (getOlapSchema()). olapConnection.setLocale(Locale.GERMANY); final Cube salesCubeGerman = olapConnection.getOlapSchema().getCubes().get("Sales"); assertEquals("Verkaufen", salesCubeGerman.getCaption()); assertEquals("Cube Verkaufen", salesCubeGerman.getDescription()); olapConnection.setLocale(Locale.FRANCE); final Cube salesCubeFrance = olapConnection.getOlapSchema().getCubes().get("Sales"); assertEquals("Ventes", salesCubeFrance.getCaption()); assertEquals("Cube des ventes", salesCubeFrance.getDescription()); // The sales cube queried when the connection had a different // locale has not been updated. According to the olap4j spec, // behavior is undefined (e.g. the US sales cube might be invalid). // In the xmla-olap4j driver, it stays valid but still shows the // caption under the US locale. assertEquals("Sales", salesCubeUs.getCaption()); // Reset locale. olapConnection.setLocale(Locale.US); } // Try to set locale to null, should get error. try { olapConnection.setLocale(null); fail("expected exception"); } catch (IllegalArgumentException e) { // Set if back olapConnection.setLocale(Locale.getDefault()); } // Get, set role, get available role names. final String s = olapConnection.getRoleName(); // ok if s is null olapConnection.setRoleName(null); assertNull(olapConnection.getRoleName()); olapConnection.setRoleName(s); // ok if role names list is null final List<String> roleNames = olapConnection.getAvailableRoleNames(); if (roleNames != null && s != null) { assertTrue( "role name " + s + " should be in available role names " + roleNames, roleNames.contains(s)); } // Unwrap the mondrian connection. switch (tester.getFlavor()) { case MONDRIAN: // mondrian.olap.Connection does not extend java.sql.Connection // but we should be able to unwrap it regardless final Class<?> mondrianConnectionClass; try { mondrianConnectionClass = Class.forName("mondrian.olap.Connection"); } catch (ClassNotFoundException e) { throw new RuntimeException(e); } final Object mondrianConnection = ((OlapWrapper) connection).unwrap( mondrianConnectionClass); assertNotNull(mondrianConnection); assert mondrianConnectionClass.isInstance(mondrianConnection); } } public void testXmlaCatalogParameter() throws Exception { if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() == Tester.Flavor.REMOTE_XMLA) { // We won't use the tester itself since we want to test // creating a connection with and without a Catalog parameter. Properties info = new Properties(); connection = DriverManager.getConnection( tester.getURL().replaceFirst("\\;Catalog=FoodMart", ""), info); assertEquals("FoodMart", connection.getCatalog()); final NamedList<Catalog> catalogs = ((OlapConnection) connection).getOlapCatalogs(); assertNotNull(catalogs); Statement statement = connection.createStatement(); OlapStatement olapStatement = TestContext.Wrapper.NONE.unwrap(statement, OlapStatement.class); assertSame(connection, olapStatement.getConnection()); CellSet cellSet = olapStatement.executeOlapQuery( "SELECT FROM [Sales]"); assertSame(statement, cellSet.getStatement()); List<CellSetAxis> axesList = cellSet.getAxes(); assertNotNull(axesList); assertEquals(0, axesList.size()); info.setProperty( XmlaOlap4jDriver.Property.CATALOG.name(), "FoodMart"); connection = DriverManager.getConnection( tester.getURL().replaceFirst("\\;Catalog=FoodMart", ""), info); assertEquals("FoodMart", connection.getCatalog()); info.setProperty( XmlaOlap4jDriver.Property.CATALOG.name(), "FoodMartError"); try { connection = DriverManager.getConnection( tester.getURL().replaceFirst("\\;Catalog=FoodMart", ""), info); Statement statement2 = connection.createStatement(); OlapStatement olapStatement2 = TestContext.Wrapper.NONE.unwrap( statement2, OlapStatement.class); olapStatement2.executeOlapQuery( "SELECT FROM [Sales]"); fail(); } catch (OlapException e) { // no op. } } } public void testDatabaseMetaData() throws SQLException { connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final OlapDatabaseMetaData databaseMetaData = olapConnection.getMetaData(); // as per testDriver Driver driver = DriverManager.getDriver(tester.getDriverUrlPrefix()); assertNotNull(driver); assertEquals( databaseMetaData.getDriverMajorVersion(), driver.getMajorVersion()); assertEquals( databaseMetaData.getDriverMinorVersion(), driver.getMinorVersion()); final String driverName = databaseMetaData.getDriverName(); // typically a string like "Mondrian olap4j driver" assertTrue( driverName != null && driverName.length() > 10); final String driverVersion = databaseMetaData.getDriverVersion(); // typically a string like "0.9" or "3.1.2" assertTrue( driverVersion != null && driverName.length() > 2); } public void testStatement() throws SQLException { connection = tester.createConnection(); Statement statement = connection.createStatement(); // Closing a statement is idempotent. assertIsClosed(statement, false); statement.close(); assertIsClosed(statement, true); statement.close(); assertIsClosed(statement, true); // Unwrap the statement to get the olap statement. Depending on the // driver, this may or may not be the same object. statement = connection.createStatement(); OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); assertNotNull(olapStatement); // Execute a simple query. CellSet cellSet = olapStatement.executeOlapQuery( "SELECT FROM [rdfh-inst:dsd]"); List<CellSetAxis> axesList = cellSet.getAxes(); assertNotNull(axesList); assertEquals(0, axesList.size()); // Executing another query implicitly closes the previous result set. assertIsClosed(statement, false); assertIsClosed(cellSet, false); CellSet cellSet2 = olapStatement.executeOlapQuery( "SELECT FROM [rdfh-inst:dsd]"); assertIsClosed(statement, false); assertIsClosed(cellSet, true); // Close the statement; this closes the result set. assertIsClosed(cellSet2, false); statement.close(); assertIsClosed(statement, true); assertIsClosed(cellSet2, true); cellSet.close(); assertIsClosed(statement, true); assertIsClosed(cellSet2, true); assertIsClosed(cellSet, true); // Close the connection. connection.close(); } public void testAxes() throws SQLException { connection = tester.createConnection(); Statement statement = connection.createStatement(); OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); CellSet cellSet = olapStatement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Store].Children} on 1\n" + "FROM [Sales]\n" + "WHERE ([Time].[1997].[Q1], [Gender].[F])"); List<CellSetAxis> axesList = cellSet.getAxes(); assertEquals(2, axesList.size()); final Member rowsMember = axesList.get(0).getPositions().get(0).getMembers().get(0); assertTrue( rowsMember.getUniqueName(), rowsMember instanceof Measure); final Member columnsMember = axesList.get(1).getPositions().get(0).getMembers().get(0); assertTrue( columnsMember.getUniqueName(), !(columnsMember instanceof Measure)); // filter axis final CellSetAxis filterAxis = cellSet.getFilterAxis(); assertEquals(1, filterAxis.getPositionCount()); final List<Position> positions = filterAxis.getPositions(); assertEquals(1, positions.size()); final Position pos0 = positions.get(0); assertEquals(0, pos0.getOrdinal()); // The filter contains members explicitly returned. It does not contain // the members of hierarchies not mentioned on any axis (e.g. // [Marital Status]). assertEquals(2, pos0.getMembers().size()); assertEquals("Q1", pos0.getMembers().get(0).getName()); assertEquals("F", pos0.getMembers().get(1).getName()); // If the query has no WHERE clause, the CellSet has a filter axis, but // it has one position and zero members (i.e. equivalent to "WHERE ()".) CellSet cellSetNoSlicer = olapStatement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Store].Children} on 1\n" + "FROM [Sales]"); assertEquals(2, cellSetNoSlicer.getAxes().size()); final CellSetAxis filterAxisNoSlicer = cellSetNoSlicer.getFilterAxis(); assertNotNull(filterAxisNoSlicer); assertEquals(1, filterAxisNoSlicer.getPositionCount()); final Position position = filterAxisNoSlicer.getPositions().get(0); assertEquals(0, position.getMembers().size()); } /** * Tests a filter with more than one position. * * @throws SQLException on error */ public void testCompoundFilter() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); OlapStatement statement = olapConnection.createStatement(); CellSet cellSet = statement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Product].Children} on 1\n" + "FROM [Sales]\n" + "WHERE [Time].[1997].[Q1] * [Gender].Members"); List<CellSetAxis> axesList = cellSet.getAxes(); assertEquals(2, axesList.size()); final CellSetAxis filterAxis = cellSet.getFilterAxis(); assertEquals(3, filterAxis.getPositionCount()); final List<Position> filterPositions = filterAxis.getPositions(); assertEquals(3, filterPositions.size()); final Position filterPosition = filterPositions.get(2); assertEquals(2, filterPosition.getMembers().size()); assertEquals("M", filterPosition.getMembers().get(1).getName()); } /** * Tests a filter with zero positions. * * @throws SQLException on error */ public void testEmptyFilter() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); OlapStatement statement = olapConnection.createStatement(); CellSet cellSet = statement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Product].Children} on 1\n" + "FROM [Sales]\n" + "WHERE [Time].[1997].[Q1] * [Gender].Parent"); List<CellSetAxis> axesList = cellSet.getAxes(); assertEquals(2, axesList.size()); final CellSetAxis filterAxis = cellSet.getFilterAxis(); assertEquals(0, filterAxis.getPositionCount()); final List<Position> filterPositions = filterAxis.getPositions(); assertEquals(0, filterPositions.size()); assertEquals(2, filterAxis.getAxisMetaData().getHierarchies().size()); final Cell cell = cellSet.getCell(Arrays.asList(0, 0)); assertTrue(cell.isNull()); } /** * Tests a query with no filter (no WHERE clause). * * @throws SQLException on error */ public void testMissingFilter() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); OlapStatement statement = olapConnection.createStatement(); CellSet cellSet = statement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Product].Children} on 1\n" + "FROM [Sales]\n"); List<CellSetAxis> axesList = cellSet.getAxes(); assertEquals(2, axesList.size()); final CellSetAxis filterAxis = cellSet.getFilterAxis(); assertEquals(1, filterAxis.getPositionCount()); final List<Position> filterPositions = filterAxis.getPositions(); assertEquals(1, filterPositions.size()); final Position position = filterPositions.get(0); assertEquals(0, position.getMembers().size()); assertEquals( 0, filterAxis.getAxisMetaData().getHierarchies().size()); assertTrue(filterAxis.getAxisMetaData().getHierarchies().isEmpty()); final Cell cell = cellSet.getCell(Arrays.asList(0, 0)); assertEquals("24,597", cell.getFormattedValue()); } public void testMeasureVersusMemberCasting() throws Exception { connection = tester.createConnection(); Statement statement = connection.createStatement(); OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); CellSet cellSet = olapStatement.executeOlapQuery( "SELECT {[Measures].[Unit Sales]} on 0,\n" + "{[Store].Children} on 1\n" + "FROM [Sales]\n" + "WHERE ([Time].[1997].[Q1], [Gender].[F])"); List<CellSetAxis> axesList = cellSet.getAxes(); assertEquals(2, axesList.size()); final Member rowsMember = axesList.get(0).getPositions().get(0).getMembers().get(0); assertTrue( rowsMember.getUniqueName(), rowsMember instanceof Measure); final Member columnsMember = axesList.get(1).getPositions().get(0).getMembers().get(0); assertTrue( columnsMember.getUniqueName(), !(columnsMember instanceof Measure)); } public void testInvalidStatement() throws SQLException { connection = tester.createConnection(); Statement statement = connection.createStatement(); OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); // Execute a query with a syntax error. try { CellSet cellSet = olapStatement.executeOlapQuery( "SELECT an error FROM [Sales]"); fail("expected error, got " + cellSet); } catch (OlapException e) { switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: assertTrue( e.getMessage(), e.getMessage().indexOf("XMLA MDX parse failed") >= 0); break; default: assertTrue( TestContext.getStackTrace(e).indexOf( "Failed to parse query") >= 0); break; } } // Error does not cause statement to become closed. assertIsClosed(olapStatement, false); olapStatement.close(); connection.close(); } private enum Method { ClassName, Mode, Type, TypeName, OlapType } public void testPreparedStatement() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); PreparedOlapStatement pstmt = olapConnection.prepareOlapStatement( "SELECT {\n" + " Parameter(\"P1\", [Store], [Store].[USA].[CA]).Parent,\n" + " ParamRef(\"P1\").Children} ON 0\n" + "FROM [Sales]\n" + "WHERE [Gender].[M]"); OlapParameterMetaData parameterMetaData = pstmt.getParameterMetaData(); int paramCount = parameterMetaData.getParameterCount(); // XMLA driver does not support parameters yet. switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: assertEquals(0, paramCount); return; } // Mondrian driver supports parameters. assertEquals(1, paramCount); int[] paramIndexes = {0, 1, 2}; for (int paramIndex : paramIndexes) { for (Method method : Method.values()) { try { switch (method) { case ClassName: String className = parameterMetaData.getParameterClassName(paramIndex); assertEquals("org.olap4j.metadata.Member", className); break; case Mode: int mode = parameterMetaData.getParameterMode(paramIndex); assertEquals(ParameterMetaData.parameterModeIn, mode); break; case Type: int type = parameterMetaData.getParameterType(paramIndex); assertEquals(Types.OTHER, type); break; case TypeName: String typeName = parameterMetaData.getParameterTypeName(paramIndex); assertEquals( "MemberType<hierarchy=[Store].[Store]>", typeName); break; case OlapType: Type olapType = parameterMetaData.getParameterOlapType(paramIndex); assertEquals( "MemberType<hierarchy=[Store].[Store]>", olapType.toString()); break; default: throw Olap4jUtil.unexpected(method); } if (paramIndex != 1) { fail("expected exception"); } } catch (SQLException e) { if (paramIndex == 1) { throw e; } else { // ok - expecting exception } } } } if (tester.getFlavor() == Tester.Flavor.XMLA) // TODO: upgrade mondrian assertEquals("Sales", pstmt.getCube().getName()); // Check metadata exists. (Support for this method is optional.) final CellSetMetaData metaData = pstmt.getMetaData(); assertEquals("Sales", metaData.getCube().getName()); CellSet cellSet = pstmt.executeQuery(); assertEquals(metaData, cellSet.getMetaData()); assertEquals("Sales", metaData.getCube().getName()); String s = TestContext.toString(cellSet); final String expected = "Axis #0:\n" + "{[Gender].[Gender].[M]}\n" + "Axis #1:\n" + "{[Store].[Store].[USA]}\n" + "{[Store].[Store].[USA].[CA].[Alameda]}\n" + "{[Store].[Store].[USA].[CA].[Beverly Hills]}\n" + "{[Store].[Store].[USA].[CA].[Los Angeles]}\n" + "{[Store].[Store].[USA].[CA].[San Diego]}\n" + "{[Store].[Store].[USA].[CA].[San Francisco]}\n" + "Row #0: 135,215\n" + "Row #0: \n" + "Row #0: 10,562\n" + "Row #0: 13,574\n" + "Row #0: 12,800\n" + "Row #0: 1,053\n"; TestContext.assertEqualsVerbose(expected, s); // Bind parameter and re-execute. final List<Position> positions = cellSet.getAxes().get(0).getPositions(); final Member member = positions.get(positions.size() - 1).getMembers().get(0); assertFalse(pstmt.isSet(1)); // parameter is 'set' even if value is null pstmt.setObject(1, null); assertTrue(pstmt.isSet(1)); pstmt.unset(1); assertFalse(pstmt.isSet(1)); pstmt.setObject(1, member); assertTrue(pstmt.isSet(1)); CellSet cellSet2 = pstmt.executeQuery(); assertIsClosed(cellSet, true); assertIsClosed(cellSet2, false); s = TestContext.toString(cellSet2); final String expected2 = "Axis #0:\n" + "{[Gender].[Gender].[M]}\n" + "Axis #1:\n" + "{[Store].[Store].[USA].[CA]}\n" + "{[Store].[Store].[USA].[CA].[San Francisco].[Store 14]}\n" + "Row #0: 37,989\n" + "Row #0: 1,053\n"; TestContext.assertEqualsVerbose(expected2, s); // Unset parameter and re-execute. pstmt.unset(1); cellSet = pstmt.executeQuery(); s = TestContext.toString(cellSet); TestContext.assertEqualsVerbose(expected, s); // Re-execute with a new MDX string. CellSet cellSet3 = pstmt.executeOlapQuery( "SELECT FROM [Sales] WHERE [Time].[Weekly].[1997].[3]"); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{[Time].[Weekly].[1997].[3]}\n" + "9,518", TestContext.toString(cellSet3)); // Number of parameters has changed. OlapParameterMetaData parameterMetaData1 = pstmt.getParameterMetaData(); assertEquals(0, parameterMetaData1.getParameterCount()); // Try to bind non-existent parameter. try { pstmt.setInt(1, 100); fail("expected exception"); } catch (SQLException e) { // ok } // Execute again. CellSet cellSet4 = pstmt.executeQuery(); assertIsClosed(cellSet4, false); assertIsClosed(cellSet3, true); assertEquals(0, cellSet4.getAxes().size()); assertEquals(9518.0, cellSet4.getCell(0).getValue()); // Re-execute with a parse tree. MdxParser mdxParser = olapConnection.getParserFactory().createMdxParser(olapConnection); SelectNode select = mdxParser.parseSelect( "select {[Gender]} on columns from [sales]\n" + "where [Time].[Time].[1997].[Q4]"); CellSet cellSet5 = pstmt.executeOlapQuery(select); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{[Time].[Time].[1997].[Q4]}\n" + "Axis #1:\n" + "{[Gender].[Gender].[All Gender]}\n" + "Row #0: 72,024\n", TestContext.toString(cellSet5)); // Execute. CellSet cellSet6 = pstmt.executeQuery(); assertIsClosed(cellSet6, false); assertIsClosed(cellSet5, true); assertEquals(1, cellSet6.getAxes().size()); assertEquals(72024.0, cellSet6.getCell(0).getDoubleValue()); // Close prepared statement. assertIsClosed(pstmt, false); pstmt.close(); assertIsClosed(pstmt, true); assertIsClosed(cellSet, true); assertIsClosed(cellSet2, true); assertIsClosed(cellSet6, true); // todo: test all of the PreparedOlapStatement.setXxx methods if (false) { pstmt.getCube(); } } public void testCellSetMetaData() throws SQLException { // Metadata of prepared statement connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); checkCellSetMetaData1( olapConnection, "select {[Gender]} on columns from [sales]\n" + "where [Time].[1997].[Q4]"); // now a query with no explicit slicer checkCellSetMetaData1( olapConnection, "select {[Gender]} on columns from [sales]"); } private void checkCellSetMetaData1( OlapConnection olapConnection, String mdx) throws SQLException { PreparedOlapStatement pstmt = olapConnection.prepareOlapStatement(mdx); final CellSetMetaData cellSetMetaData = pstmt.getMetaData(); checkCellSetMetaData(cellSetMetaData, 1, null); // Metadata of its cellset final CellSet cellSet = pstmt.executeQuery(); checkCellSetMetaData(cellSet.getMetaData(), 1, cellSet); // Metadata of regular statement executing string. final OlapStatement stmt = olapConnection.createStatement(); final CellSet cellSet1 = stmt.executeOlapQuery(mdx); checkCellSetMetaData(cellSet1.getMetaData(), 1, cellSet1); // Metadata of regular statement executing parse tree. MdxParser mdxParser = olapConnection.getParserFactory().createMdxParser(olapConnection); SelectNode select = mdxParser.parseSelect(mdx); final OlapStatement stmt2 = olapConnection.createStatement(); CellSet cellSet2 = stmt2.executeOlapQuery(select); checkCellSetMetaData(cellSet2.getMetaData(), 1, cellSet2); } private void checkCellSetMetaData( CellSetMetaData cellSetMetaData, int axesCount, CellSet cellSet) throws OlapException { assertNotNull(cellSetMetaData); assertEquals(axesCount, cellSetMetaData.getAxesMetaData().size()); assertEquals("Sales", cellSetMetaData.getCube().getName()); int k = -1; final Set<Hierarchy> unseenHierarchies = new HashSet<Hierarchy>(cellSetMetaData.getCube().getHierarchies()); for (CellSetAxisMetaData axisMetaData : cellSetMetaData.getAxesMetaData()) { ++k; assertEquals( Axis.Factory.forOrdinal(k), axisMetaData.getAxisOrdinal()); assertEquals(k, axisMetaData.getAxisOrdinal().axisOrdinal()); assertTrue(axisMetaData.getHierarchies().size() > 0); unseenHierarchies.removeAll(axisMetaData.getHierarchies()); assertTrue(axisMetaData.getProperties().size() == 0); if (cellSet != null) { final CellSetAxisMetaData cellSetAxisMetaData = cellSet.getAxes().get(k).getAxisMetaData(); assertEquals(cellSetAxisMetaData, axisMetaData); } } CellSetAxisMetaData axisMetaData = cellSetMetaData.getFilterAxisMetaData(); assertNotNull(axisMetaData); assertEquals(Axis.FILTER, axisMetaData.getAxisOrdinal()); assertTrue(axisMetaData.getHierarchies().size() >= 0); if (false) { // The slicer used to contain all hierarchies not seen on other // axes. No longer true. assertEquals( new HashSet<Hierarchy>(axisMetaData.getHierarchies()), unseenHierarchies); } else { for (Hierarchy hierarchy : axisMetaData.getHierarchies()) { assertTrue(unseenHierarchies.contains(hierarchy)); } } assertTrue(axisMetaData.getProperties().size() == 0); if (cellSet != null) { assertEquals( cellSet.getFilterAxis().getAxisMetaData(), axisMetaData); assertEquals( 1, cellSet.getFilterAxis().getPositionCount()); assertEquals( 1, cellSet.getFilterAxis().getPositions().size()); } } /** * Tests the {@link CellSetAxisMetaData} class, based on an example * in the javadoc of same class. * * @throws Exception on error */ public void testCellSetAxisMetaData() throws Exception { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final String mdx = "SELECT\n" + " {[Measures].Members} ON COLUMNS,\n" + " CrossJoin([Store].Members, [Gender].Children)\n" + " DIMENSION PROPERTIES\n" + " MEMBER_ORDINAL,\n" + " MEMBER_UNIQUE_NAME,\n" + " DISPLAY_INFO ON ROWS\n" + " FROM [Sales]"; // first via prepared statement final PreparedOlapStatement preparedStmt = olapConnection.prepareOlapStatement(mdx); final CellSetMetaData cellSetMetaData = preparedStmt.getMetaData(); checkAxisMetaData(cellSetMetaData.getAxesMetaData().get(1)); // second via directly executed statement OlapStatement olapStatement = olapConnection.createStatement(); final CellSet cellSet = olapStatement.executeOlapQuery(mdx); checkAxisMetaData(cellSet.getAxes().get(1).getAxisMetaData()); // third via metadata of direct statement checkAxisMetaData(cellSet.getMetaData().getAxesMetaData().get(1)); } private void checkAxisMetaData(CellSetAxisMetaData cellSetAxisMetaData) { final List<Hierarchy> hierarchies = cellSetAxisMetaData.getHierarchies(); assertEquals(2, hierarchies.size()); assertEquals("Store", hierarchies.get(0).getName()); assertEquals("Gender", hierarchies.get(1).getName()); final List<Property> properties = cellSetAxisMetaData.getProperties(); switch (tester.getFlavor()) { case MONDRIAN: // todo: fix mondrian driver. If there are 3 properties and 2 // hierarchies, that's 6 properties total assertEquals(3, properties.size()); break; default: assertEquals(6, properties.size()); break; } assertEquals("MEMBER_ORDINAL", properties.get(0).getName()); assertEquals("MEMBER_UNIQUE_NAME", properties.get(1).getName()); assertEquals("DISPLAY_INFO", properties.get(2).getName()); } public void testCellSet() throws SQLException { connection = tester.createConnection(); Statement statement = connection.createStatement(); final OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); final CellSet cellSet = olapStatement.executeOlapQuery( "SELECT\n" + " {[Measures].[Unit Sales],\n" + " [Measures].[Store Sales]} ON COLUMNS\n," + " Crossjoin({[Gender].[M]}, [Product].Children) ON ROWS\n" + "FROM [Sales]\n" + "WHERE [Time].[1997].[Q2]"); String s = TestContext.toString(cellSet); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{[Time].[Time].[1997].[Q2]}\n" + "Axis #1:\n" + "{[Measures].[Unit Sales]}\n" + "{[Measures].[Store Sales]}\n" + "Axis #2:\n" + "{[Gender].[Gender].[M], [Product].[Product].[Drink]}\n" + "{[Gender].[Gender].[M], [Product].[Product].[Food]}\n" + "{[Gender].[Gender].[M], [Product].[Product].[Non-Consumable]}\n" + "Row #0: 3,023\n" + "Row #0: 6,004.80\n" + "Row #1: 22,558\n" + "Row #1: 47,869.17\n" + "Row #2: 6,037\n" + "Row #2: 12,935.16\n", s); } public void testCell() throws Exception { connection = tester.createConnection(); Statement statement = connection.createStatement(); final OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); CellSet cellSet = olapStatement.executeOlapQuery( "SELECT\n" + " {[Measures].[Unit Sales],\n" + " [Measures].[Store Sales]} ON COLUMNS\n," + " Crossjoin({[Gender].[M]}, [Product].Children) ON ROWS\n" + "FROM [Sales]\n" + "WHERE [Time].[1997].[Q2]"); // cell column#1, row#2 // cellOrdinal = colOrdinal + rowOrdinal * columnCount // = 1 + 2 * 2 // = 5 // access method 1 Cell cell = cellSet.getCell(5); assertEquals(5, cell.getOrdinal()); if (tester.getFlavor() != TestContext.Tester.Flavor.XMLA // FIXME || tester.getFlavor() != TestContext.Tester.Flavor.REMOTE_XMLA) { assertEquals(12935.16, cell.getValue()); } assertEquals(12935.16, cell.getDoubleValue()); assertEquals("12,935.16", cell.getFormattedValue()); assertEquals(cellSet, cell.getCellSet()); // access method 2 cell = cellSet.getCell(Arrays.asList(1, 2)); assertEquals(5, cell.getOrdinal()); // access method 3 cell = cellSet.getCell( cellSet.getAxes().get(0).getPositions().get(1), cellSet.getAxes().get(1).getPositions().get(2)); assertEquals(5, cell.getOrdinal()); assertEquals(Arrays.asList(1, 2), cell.getCoordinateList()); assertEquals( "#,###.00", cell.getPropertyValue(Property.StandardCellProperty.FORMAT_STRING)); assertFalse(cell.isEmpty()); assertFalse(cell.isError()); assertFalse(cell.isNull()); assertNull(cell.getErrorText()); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: // TODO: implement drill-through in XMLA driver break; default: final ResultSet resultSet = cell.drillThrough(); try { final ResultSetMetaData metaData = resultSet.getMetaData(); // Most databases return 5 columns. Derby returns 9 because of // 4 columns in the ORDER BY clause. assertTrue(metaData.getColumnCount() >= 5); assertEquals("Quarter", metaData.getColumnLabel(1)); assertEquals("Quarter_0", metaData.getColumnLabel(2)); assertEquals("Month", metaData.getColumnLabel(3)); assertEquals("Store Sales", metaData.getColumnLabel(11)); } finally { resultSet.close(); } break; } // cell out of range using getCell(int) try { Cell cell2 = cellSet.getCell(-5); fail("expected exception, got " + cell2); } catch (IndexOutOfBoundsException e) { // ok } // cell out of range using getCell(int) try { Cell cell2 = cellSet.getCell(105); fail("expected exception, got " + cell2); } catch (IndexOutOfBoundsException e) { // ok } // cell out of range using getCell(List<Integer>) try { Cell cell2 = cellSet.getCell(Arrays.asList(2, 1)); fail("expected exception, got " + cell2); } catch (IndexOutOfBoundsException e) { // ok } // cell out of range using getCell(Position...) is not possible; but // number of positions might be wrong try { // too few dimensions Cell cell2 = cellSet.getCell(cellSet.getAxes().get(0).getPositions().get(0)); fail("expected exception, got " + cell2); } catch (IllegalArgumentException e) { // ok } try { // too many dimensions Cell cell2 = cellSet.getCell( cellSet.getAxes().get(0).getPositions().get(0), cellSet.getAxes().get(1).getPositions().get(0), cellSet.getAxes().get(0).getPositions().get(0)); fail("expected exception, got " + cell2); } catch (IllegalArgumentException e) { // ok } // We provide positions from the wrong axes, but the provider doesn't // notice that they're wrong. That's OK. cell = cellSet.getCell( cellSet.getAxes().get(1).getPositions().get(1), cellSet.getAxes().get(0).getPositions().get(1)); assertEquals(3, cell.getOrdinal()); // Null cell cellSet = olapStatement.executeOlapQuery( "with member [Measures].[X] as 'IIF([Measures].[Store Sales]>10000,[Measures].[Store Sales],Null)'\n" + "select\n" + "{[Measures].[X]} on columns,\n" + "{[Product].[Product Department].members} on rows\n" + "from Sales"); cell = cellSet.getCell(0); assertFalse(cell.isNull()); cell = cellSet.getCell(2); assertTrue(cell.isNull()); // Empty cell cellSet = olapStatement.executeOlapQuery( "select from [Sales]\n" + "where ([Time].[1997].[Q4].[12],\n" + " [Product].[All Products].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Portsmouth].[Portsmouth Imported Beer],\n" + " [Store].[All Stores].[USA].[WA].[Bellingham])"); cell = cellSet.getCell(0); assertTrue(cell.isEmpty()); // Error cell cellSet = olapStatement.executeOlapQuery( "with member [Measures].[Foo] as ' Dimensions(-1).Name '\n" + "select {[Measures].[Foo]} on columns from [Sales]"); cell = cellSet.getCell(0); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: // FIXME: mondrian's XMLA provider doesn't indicate that a cell is // an error break; default: assertTrue(cell.isError()); assertEquals("Index '-1' out of bounds", cell.getErrorText()); break; } // todo: test CellSetAxis methods // // public int getAxisOrdinal() // public CellSet getCellSet() // public CellSetAxisMetaData getAxisMetaData() // public List<Position> getPositions() // public int getPositionCount() // public ListIterator<Position> iterate() // // todo: test OlapResultAxisMetaData methods // // public org.olap4j.Axis getAxisOrdinal() // public List<Hierarchy> getHierarchies() // public List<Property> getProperties() } /** * Tests different scrolling characteristics. * * <p>In one mode, you request that you get all of the positions on an axis. * You can call {@link CellSetAxis#getPositions()} and * {@link CellSetAxis#getPositions()}. * * <p>In another mode, you can iterate over the positions, calling * {@link org.olap4j.CellSetAxis#iterator()}. Note that this method returns * a {@link java.util.ListIterator}, which has * {@link java.util.ListIterator#nextIndex()}. We could maybe extend this * interface further, to allow to jump forwards/backwards by N. * * <p>This test should check that queries work correctly when you ask * for axes as lists and iterators. If you open the query as a list, * you can get it as an iterator, but if you open it as an iterator, you * cannot get it as a list. (Or maybe you can, but it is expensive.) * * <p>The existing JDBC method {@link Connection#createStatement(int, int)}, * where the 2nd parameter has values such as * {@link ResultSet#TYPE_FORWARD_ONLY} and * {@link ResultSet#TYPE_SCROLL_INSENSITIVE}, might be the way to invoke * those two modes. * * <p>Note that cell ordinals are not well-defined until axis lengths are * known. Test that cell ordinal property is not available if the statement * is opened in this mode. * * <p>It was proposed that there would be an API to get blocks of cell * values. Not in the spec yet. */ public void testScrolling() { // todo: submit a query where you ask for different scrolling // characteristics. maybe the values int x = ResultSet.TYPE_SCROLL_INSENSITIVE; int y = ResultSet.TYPE_FORWARD_ONLY; // are how to ask for these characteristics. also need to document this // behavior in the API and the spec. in one mode, } /** * Tests creation of an MDX parser, and converting an MDX statement into * a parse tree. * * <p>Also create a set of negative tests. Do we give sensible errors if * the MDX is misformed. * * <p>Also have a set of validator tests, which check that the functions * used exist, are applied to arguments of the correct type, and members * exist. */ public void testParsing() throws SQLException { // parse connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); MdxParser mdxParser = olapConnection.getParserFactory().createMdxParser(olapConnection); SelectNode select = mdxParser.parseSelect( "with member [Measures].[Foo] as ' [Measures].[Bar] ', FORMAT_STRING='xxx'\n" + " select {[Gender]} on columns, {[Store].Children} on rows\n" + "from [sales]\n" + "where [Time].[1997].[Q4]"); // unparse checkUnparsedMdx(select); // test that get error if axes do not have unique names select = mdxParser.parseSelect( "select {[Gender]} on columns, {[Store].Children} on columns\n" + "from [sales]"); if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() != Tester.Flavor.XMLA) { // This test requires validator support. return; } MdxValidator validator = olapConnection.getParserFactory().createMdxValidator( olapConnection); try { select = validator.validateSelect(select); fail("expected exception, got " + select); } catch (Exception e) { assertTrue( TestContext.getStackTrace(e) .indexOf("Duplicate axis name 'COLUMNS'.") >= 0); } } private void checkUnparsedMdx(SelectNode select) { checkUnparsedMdx( select, "WITH\n" + "MEMBER [Measures].[Foo] AS\n" + " [Measures].[Bar], FORMAT_STRING = \"xxx\"\n" + "SELECT\n" + "{[Gender]} ON COLUMNS,\n" + "{[Store].Children} ON ROWS\n" + "FROM [sales]\n" + "WHERE [Time].[1997].[Q4]"); } private void checkUnparsedMdx( SelectNode select, String expectedMdx) { StringWriter sw = new StringWriter(); ParseTreeWriter parseTreeWriter = new ParseTreeWriter(sw); select.unparse(parseTreeWriter); String mdx = sw.toString(); TestContext.assertEqualsVerbose( expectedMdx, mdx); } /** * Tests creation of an MDX query from a parse tree. Build the parse tree * programmatically. */ public void testUnparsing() { // Note that the select statement constructed here is equivalent // to the one in testParsing. final IdentifierNode cubeName = new IdentifierNode(new NameSegment("sales")); SelectNode select = new SelectNode( null, new ArrayList<ParseTreeNode>(), new ArrayList<AxisNode>(), cubeName, new AxisNode( null, false, Axis.FILTER, new ArrayList<IdentifierNode>(), null), new ArrayList<IdentifierNode>()); select.getWithList().add( new WithMemberNode( null, new IdentifierNode( new NameSegment("Measures"), new NameSegment("Foo")), new IdentifierNode( new NameSegment("Measures"), new NameSegment("Bar")), Arrays.asList( new PropertyValueNode( null, "FORMAT_STRING", LiteralNode.createString( null, "xxx"))))); select.getAxisList().add( new AxisNode( null, false, Axis.COLUMNS, new ArrayList<IdentifierNode>(), new CallNode( null, "{}", Syntax.Braces, Arrays.asList( (ParseTreeNode) new IdentifierNode( new NameSegment("Gender")))))); select.getAxisList().add( new AxisNode( null, false, Axis.ROWS, new ArrayList<IdentifierNode>(), new CallNode( null, "{}", Syntax.Braces, new CallNode( null, "Children", Syntax.Property, new IdentifierNode( new NameSegment("Store")))))); select.getFilterAxis().setExpression( new IdentifierNode( new NameSegment("Time"), new NameSegment("1997"), new NameSegment("Q4"))); assertEquals(select.getFrom(), cubeName); checkUnparsedMdx(select); // Now with a subquery in the FROM clause. SelectNode subSelect = new SelectNode( null, new ArrayList<ParseTreeNode>(), new ArrayList<AxisNode>(), new IdentifierNode(new NameSegment("warehouse")), new AxisNode( null, false, Axis.FILTER, new ArrayList<IdentifierNode>(), null), new ArrayList<IdentifierNode>()); select.setFrom(subSelect); assertEquals(select.getFrom(), subSelect); checkUnparsedMdx( select, "WITH\n" + "MEMBER [Measures].[Foo] AS\n" + " [Measures].[Bar], FORMAT_STRING = \"xxx\"\n" + "SELECT\n" + "{[Gender]} ON COLUMNS,\n" + "{[Store].Children} ON ROWS\n" + "FROM (\n" + " SELECT\n" + " FROM [warehouse])\n" + "WHERE [Time].[1997].[Q4]"); } public void testBuildParseTree() { // It is an error to create a select node with a filter axis whose type // is not filter try { SelectNode select = new SelectNode( null, new ArrayList<ParseTreeNode>(), new ArrayList<AxisNode>(), new IdentifierNode(new NameSegment("sales")), new AxisNode( null, false, Axis.COLUMNS, new ArrayList<IdentifierNode>(), null), new ArrayList<IdentifierNode>()); fail("expected error, got " + select); } catch (IllegalArgumentException e) { TestContext.checkThrowable(e, "Filter axis must have type FILTER"); } // Create a select node with empty filter axis. It is populated with // a filter axis whose expression is null. SelectNode select = new SelectNode( null, new ArrayList<ParseTreeNode>(), new ArrayList<AxisNode>(), new IdentifierNode(new NameSegment("sales")), new AxisNode( null, false, Axis.FILTER, new ArrayList<IdentifierNode>(), null), new ArrayList<IdentifierNode>()); final AxisNode filterAxis = select.getFilterAxis(); assertNotNull(filterAxis); assertNull(filterAxis.getExpression()); assertEquals(Axis.FILTER, filterAxis.getAxis()); // Parses to an expression with no WHERE clause checkUnparsedMdx( select, "SELECT\n" + "FROM [sales]"); // Set the filter, see if it takes. select.getFilterAxis().setExpression( new CallNode( null, "()", Syntax.Parentheses, new IdentifierNode( new NameSegment("Measures"), new NameSegment("Store Sales")), new IdentifierNode( new NameSegment("Gender"), new NameSegment("M")))); checkUnparsedMdx( select, "SELECT\n" + "FROM [sales]\n" + "WHERE ([Measures].[Store Sales], [Gender].[M])"); // Set it back to null select.getFilterAxis().setExpression(null); checkUnparsedMdx( select, "SELECT\n" + "FROM [sales]"); } /** * Tests the {@link Cube#lookupMember(java.util.List)} method. */ public void testCubeLookupMember() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Cube cube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Sales Ragged"); Member member = cube.lookupMember(nameList("Time", "1997", "Q2")); assertEquals("[Time].[Time].[1997].[Q2]", member.getUniqueName()); // Member.getChildMemberCount assertEquals(3, member.getChildMemberCount()); // Member.getChildMembers final NamedList<? extends Member> childMembers = member.getChildMembers(); assertEquals(3, childMembers.size()); assertEquals( "[Time].[Time].[1997].[Q2].[4]", childMembers.get(0).getUniqueName()); assertEquals(0, childMembers.get(0).getChildMemberCount()); assertEquals( "[Time].[Time].[1997].[Q2].[6]", childMembers.get("6").getUniqueName()); assertNull(childMembers.get("1")); member = cube.lookupMember(nameList("Time", "1997", "Q5")); assertNull(member); // arguably this should return [Customers].[All Customers]; but it // makes a bit more sense for it to return null member = cube.lookupMember(nameList("Customers")); assertNull(member); member = cube.lookupMember(nameList("Customers", "All Customers")); assertTrue(member.isAll()); } /** * Tests the {@link Cube#lookupMembers(java.util.Set, java.util.List)} * method. */ public void testCubeLookupMembers() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Cube cube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Sales"); List<Member> memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.ANCESTORS, Member.TreeOp.CHILDREN), nameList("Time", "1997", "Q2")); String expected; switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: // TODO: Fix mondrian's XMLA driver to return members ordered by // level then by ordinal as per XMLA spec expected = "[Time].[1997].[Q2].[4]\n" + "[Time].[1997].[Q2].[5]\n" + "[Time].[1997].[Q2].[6]\n" + "[Time].[1997]\n"; break; default: expected = "[Time].[Time].[1997]\n" + "[Time].[Time].[1997].[Q2].[4]\n" + "[Time].[Time].[1997].[Q2].[5]\n" + "[Time].[Time].[1997].[Q2].[6]\n"; } TestContext.assertEqualsVerbose( expected, memberListToString(memberList)); // ask for non-existent member; list should be empty memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.ANCESTORS, Member.TreeOp.CHILDREN), nameList("Time", "1997", "Q5")); assertTrue(memberList.isEmpty()); // ask for parent & ancestors; should not get duplicates memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.ANCESTORS, Member.TreeOp.PARENT), nameList("Time", "1997", "Q2")); TestContext.assertEqualsVerbose( "[Time].[Time].[1997]\n", memberListToString(memberList)); // ask for parent of root member, should not get null member in list memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.ANCESTORS, Member.TreeOp.PARENT), nameList("Product")); assertTrue(memberList.isEmpty()); // ask for siblings and children, and the results should be // hierarchically ordered (as always) memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.SIBLINGS, Member.TreeOp.CHILDREN), nameList("Time", "1997", "Q2")); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: // TODO: fix mondrian's XMLA driver to return members ordered by // level then ordinal expected = "[Time].[Time].[1997].[Q2].[4]\n" + "[Time].[Time].[1997].[Q2].[5]\n" + "[Time].[Time].[1997].[Q2].[6]\n" + "[Time].[Time].[1997].[Q1]\n" + "[Time].[Time].[1997].[Q3]\n" + "[Time].[Time].[1997].[Q4]\n"; break; default: expected = "[Time].[Time].[1997].[Q1]\n" + "[Time].[Time].[1997].[Q2].[4]\n" + "[Time].[Time].[1997].[Q2].[5]\n" + "[Time].[Time].[1997].[Q2].[6]\n" + "[Time].[Time].[1997].[Q3]\n" + "[Time].[Time].[1997].[Q4]\n"; break; } TestContext.assertEqualsVerbose( expected, memberListToString(memberList)); // siblings of the root member - potentially tricky memberList = cube.lookupMembers( Olap4jUtil.enumSetOf(Member.TreeOp.SIBLINGS), nameList("Time", "1997")); TestContext.assertEqualsVerbose( "[Time].[Time].[1998]\n", memberListToString(memberList)); memberList = cube.lookupMembers( Olap4jUtil.enumSetOf( Member.TreeOp.SIBLINGS, Member.TreeOp.SELF), nameList("Customers", "USA", "OR")); TestContext.assertEqualsVerbose( "[Customers].[Customers].[USA].[CA]\n" + "[Customers].[Customers].[USA].[OR]\n" + "[Customers].[Customers].[USA].[WA]\n", memberListToString(memberList)); } /** * Tests metadata browsing. */ public void testMetadata() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); OlapDatabaseMetaData metadata = olapConnection.getMetaData(); // We engineered the XMLA test environment to have two catalogs. switch (tester.getFlavor()) { case REMOTE_XMLA: assertEquals(1, olapConnection.getOlapCatalogs().size()); break; case XMLA: assertEquals(2, olapConnection.getOlapCatalogs().size()); break; case MONDRIAN: assertEquals(1, olapConnection.getOlapCatalogs().size()); break; } Cube cube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Sales"); int z = 0; int hierarchyCount = 0; for (Dimension dimension : cube.getDimensions()) { // Call every method of Dimension assertNotNull(dimension.getCaption()); dimension.getDescription(); assertNotNull(dimension.getDefaultHierarchy()); assertEquals( dimension.getName().equals("Time") ? Dimension.Type.TIME : dimension.getName().equals("Measures") ? Dimension.Type.MEASURE : Dimension.Type.OTHER, dimension.getDimensionType()); assertNotNull(dimension.getName()); assertNotNull(dimension.getUniqueName()); for (Hierarchy hierarchy : dimension.getHierarchies()) { ++hierarchyCount; // Call every method of Hierarchy final NamedList<Member> rootMemberList = hierarchy.getRootMembers(); if (hierarchy.hasAll()) { assertEquals(1, rootMemberList.size()); } for (Member rootMember : rootMemberList) { assertNull(rootMember.getParentMember()); } assertEquals( rootMemberList, hierarchy.getLevels().get(0).getMembers()); assertNotNull(hierarchy.getDefaultMember()); assertNotNull(hierarchy.getName()); assertNotNull(hierarchy.getUniqueName()); hierarchy.getDescription(); assertNotNull(hierarchy.getCaption()); assertEquals(dimension, hierarchy.getDimension()); for (Level level : hierarchy.getLevels()) { if (level.getCardinality() >= 100) { continue; } if (level.getName().equals("Year")) { assertEquals( Level.Type.TIME_YEARS, level.getLevelType()); assertFalse(level.isCalculated()); ++z; } if (level.getName().equals("Gender")) { assertEquals(Level.Type.REGULAR, level.getLevelType()); assertFalse(level.isCalculated()); ++z; } if (level.getName().equals("Measures")) { assertEquals(Level.Type.REGULAR, level.getLevelType()); assertFalse(level.isCalculated()); ++z; } // for (Member member : level.getMembers()) { // assertNotNull(member.getName()); // assertEquals(level, member.getLevel()); // if (dimension.getDimensionType() // == Dimension.Type.MEASURE) // { // assertTrue(member instanceof Measure); // } // if (++k > 3) { // break; // } // } } } } // Make sure every hierarchy which came out through // cube.getDimensions().getHierarchies() also comes out through // cube.getHierarchies(). for (Hierarchy hierarchy : cube.getHierarchies()) { --hierarchyCount; assertNotNull(hierarchy.getName()); } assertEquals(0, hierarchyCount); assertEquals("found Year, Measures, Gender levels", 3, z); // Look for the Time.Weekly hierarchy, the 2nd hierarchy in the Time // dimension. final Hierarchy timeWeeklyHierarchy = cube.getHierarchies().get("Weekly"); assertNotNull(timeWeeklyHierarchy); assertEquals("Time", timeWeeklyHierarchy.getDimension().getName()); assertEquals( 2, timeWeeklyHierarchy.getDimension().getHierarchies().size()); Cube warehouseCube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Warehouse"); int count = 0; for (NamedSet namedSet : warehouseCube.getSets()) { ++count; assertNotNull(namedSet.getName()); assertNotNull(namedSet.getUniqueName()); assertNotNull(namedSet.getCaption()); namedSet.getDescription(); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: // FIXME: implement getExpression in XMLA driver break; default: assertTrue( namedSet.getExpression().getType() instanceof SetType); } } assertTrue(count > 0); // ~ Member Member member = cube.lookupMember( nameList("Product", "Food", "Marshmallows")); assertNull(member); // we don't sell marshmallows! member = cube.lookupMember( nameList("Product", "Food")); assertNotNull(member); Member member2 = cube.lookupMember( nameList("Product", "All Products", "Food")); assertEquals(member, member2); final Member bread = cube.lookupMember( nameList("Product", "Food", "Baked Goods", "Bread")); assertEquals("[Product].[Product].[Food]", member.getUniqueName()); assertEquals("Food", member.getName()); assertEquals( "[Product].[Product].[Product Family]", member.getLevel().getUniqueName()); assertEquals(Member.Type.REGULAR, member.getMemberType()); assertEquals( "[Product].[Product].[Food].[Baked Goods]", bread.getParentMember().getUniqueName()); final List<Member> list = bread.getAncestorMembers(); assertEquals(3, list.size()); assertEquals( "[Product].[Product].[Food].[Baked Goods]", list.get(0).getUniqueName()); assertEquals( "[Product].[Product].[Food]", list.get(1).getUniqueName()); assertEquals( "[Product].[Product].[All Products]", list.get(2).getUniqueName()); assertEquals("Food", member.getCaption()); switch (tester.getFlavor()) { case XMLA: case REMOTE_XMLA: assertEquals("", member.getDescription()); assertEquals(204, member.getOrdinal()); break; default: assertNull(member.getDescription()); // mondrian does not set ordinals correctly assertEquals(-1, member.getOrdinal()); assertEquals(1, member.getDepth()); assertEquals(-1, member.getSolveOrder()); assertFalse(member.isHidden()); assertNull(member.getDataMember()); assertFalse(member.isCalculatedInQuery()); } final NamedList<Property> propertyList = member.getProperties(); final Property property = propertyList.get("MEMBER_CAPTION"); assertEquals("Food", member.getPropertyFormattedValue(property)); assertEquals("Food", member.getPropertyValue(property)); assertFalse(member.isAll()); // All member final Member allProductsMember = member.getParentMember(); assertEquals( "[Product].[Product].[All Products]", allProductsMember.getUniqueName()); assertEquals("(All)", allProductsMember.getLevel().getName()); assertEquals( "[Product].[Product].[(All)]", allProductsMember.getLevel().getUniqueName()); assertEquals(1, allProductsMember.getLevel().getMembers().size()); assertTrue(allProductsMember.isAll()); assertNull(allProductsMember.getParentMember()); // ~ Property assertEquals("MEMBER_CAPTION", property.getName()); assertEquals("MEMBER_CAPTION", property.getUniqueName()); assertEquals( Olap4jUtil.enumSetOf(Property.TypeFlag.MEMBER), property.getType()); assertEquals(Datatype.STRING, property.getDatatype()); // PARENT_LEVEL property final Property parentLevelProperty = propertyList.get("PARENT_LEVEL"); assertNotNull(parentLevelProperty); assertEquals( 0, allProductsMember.getPropertyValue(parentLevelProperty)); assertEquals(0, member.getPropertyValue(parentLevelProperty)); assertEquals(2, bread.getPropertyValue(parentLevelProperty)); // PARENT_UNIQUE_NAME property final Property parentUniqueNameProperty = propertyList.get("PARENT_UNIQUE_NAME"); assertNotNull(parentUniqueNameProperty); assertNull( allProductsMember.getPropertyValue(parentUniqueNameProperty)); assertEquals( "[Product].[Product].[All Products]", member.getPropertyValue(parentUniqueNameProperty)); assertEquals( "[Product].[Product].[Food].[Baked Goods]", bread.getPropertyValue(parentUniqueNameProperty)); // Measures int k = -1; Set<String> measureNameSet = new HashSet<String>(); for (Measure measure : cube.getMeasures()) { ++k; // The first measure is [Unit Sales], because the list must be // sorted by ordinal. if (k == 0) { assertEquals("Unit Sales", measure.getName()); } if (measure.getName().equals("Profit Growth") || measure.getName().equals("Profit last Period") || measure.getName().equals("Profit")) { assertEquals(Member.Type.FORMULA, measure.getMemberType()); assertTrue(measure.isCalculated()); } else { assertEquals(Member.Type.MEASURE, measure.getMemberType()); assertFalse(measure.isCalculated()); } assertNotNull(measure.getName()); assertNotNull(measure.getAggregator()); assertTrue(measure.getDatatype() != null); // mondrian's olap4j driver returns the invisible member // [Measures].[Profit last Period]; the xmla driver does not, // because XMLA by default does not return invisible measures. if (measure.getName().equals("Profit last Period")) { assertFalse(measure.isVisible()); } else { measureNameSet.add(measure.getName()); } } assertEquals( new HashSet<String>( Arrays.asList( "Unit Sales", "Customer Count", "Profit", "Profit Growth", "Promotion Sales", "Sales Count", "Store Sales", "Store Cost")), measureNameSet); } /** * Test case for {@link org.olap4j.metadata.Schema#getSharedDimensions()}. * Bug 3375355, "getSharedDimensions returns empty result". */ public void testSchemaGetSharedDimensions() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final List<String> list = new ArrayList<String>(); final NamedList<Dimension> sharedDimensions = olapConnection.getOlapSchema().getSharedDimensions(); for (Dimension dimension : sharedDimensions) { list.add(dimension.getName()); } // note that, per specification, list is sorted assertEquals( Arrays.asList( "Product", "Store", "Store Size in SQFT", "Store Type", "Time", "Warehouse"), list); int hierarchyCount = 0, levelCount = 0; for (Dimension sharedDimension : sharedDimensions) { for (Hierarchy hierarchy : sharedDimension.getHierarchies()) { ++hierarchyCount; for (Level level : hierarchy.getLevels()) { ++levelCount; } } } assertTrue(hierarchyCount > 0); assertTrue(levelCount > 0); } /** * Testcase for bug 3312701, "VirtualCube doesn't show * Calculated Members" */ public void testVirtualCubeCmBug() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Cube warehouseSalesCube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Warehouse and Sales"); Set<String> measureVcNameSet = new HashSet<String>(); for (Measure measure : warehouseSalesCube.getMeasures()) { measureVcNameSet.add(measure.getName()); } assertEquals( new HashSet<String>( Arrays.asList( "Sales Count", "Store Cost", "Store Sales", "Unit Sales", "Profit", "Profit Growth", "Profit Per Unit Shipped", "Store Invoice", "Supply Time", "Units Ordered", "Units Shipped", "Warehouse Cost", "Warehouse Profit", "Warehouse Sales", "Average Warehouse Sale")), measureVcNameSet); } /** * Testcase for bug 1868075, "Query on ragged hierarchy gives only empty * cells". */ public void testRagged() throws SQLException { connection = tester.createConnection(); Statement statement = connection.createStatement(); final OlapStatement olapStatement = tester.getWrapper().unwrap(statement, OlapStatement.class); final CellSet cellSet = olapStatement.executeOlapQuery( "SELECT\n" + "{[Product].[All Products].[Drink].[Alcoholic Beverages].Children,\n" + "[Product].[All Products].[Food].[Baked Goods].Children} ON COLUMNS,\n" + "CrossJoin([Store].[All Stores].[USA].[CA].Children,\n" + "[Time].[1997].[Q1].Children) ON ROWS\n" + "FROM [Sales Ragged]"); String s = TestContext.toString(cellSet); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{}\n" + "Axis #1:\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine]}\n" + "{[Product].[Product].[Food].[Baked Goods].[Bread]}\n" + "Axis #2:\n" + "{[Store].[Store].[USA].[CA].[Alameda], [Time].[Time].[1997].[Q1].[1]}\n" + "{[Store].[Store].[USA].[CA].[Alameda], [Time].[Time].[1997].[Q1].[2]}\n" + "{[Store].[Store].[USA].[CA].[Alameda], [Time].[Time].[1997].[Q1].[3]}\n" + "{[Store].[Store].[USA].[CA].[Beverly Hills], [Time].[Time].[1997].[Q1].[1]}\n" + "{[Store].[Store].[USA].[CA].[Beverly Hills], [Time].[Time].[1997].[Q1].[2]}\n" + "{[Store].[Store].[USA].[CA].[Beverly Hills], [Time].[Time].[1997].[Q1].[3]}\n" + "{[Store].[Store].[USA].[CA].[Los Angeles], [Time].[Time].[1997].[Q1].[1]}\n" + "{[Store].[Store].[USA].[CA].[Los Angeles], [Time].[Time].[1997].[Q1].[2]}\n" + "{[Store].[Store].[USA].[CA].[Los Angeles], [Time].[Time].[1997].[Q1].[3]}\n" + "{[Store].[Store].[USA].[CA].[San Francisco], [Time].[Time].[1997].[Q1].[1]}\n" + "{[Store].[Store].[USA].[CA].[San Francisco], [Time].[Time].[1997].[Q1].[2]}\n" + "{[Store].[Store].[USA].[CA].[San Francisco], [Time].[Time].[1997].[Q1].[3]}\n" + "Row #0: \n" + "Row #0: \n" + "Row #1: \n" + "Row #1: \n" + "Row #2: \n" + "Row #2: \n" + "Row #3: 22\n" + "Row #3: 63\n" + "Row #4: 28\n" + "Row #4: 59\n" + "Row #5: 28\n" + "Row #5: 39\n" + "Row #6: 70\n" + "Row #6: 51\n" + "Row #7: 89\n" + "Row #7: 51\n" + "Row #8: 27\n" + "Row #8: 54\n" + "Row #9: 6\n" + "Row #9: 2\n" + "Row #10: 3\n" + "Row #10: 7\n" + "Row #11: 2\n" + "Row #11: 10\n", s); } /** * Tests members from a parent-child hierarchy. * * @throws ClassNotFoundException * @throws SQLException */ public void testParentChild() throws ClassNotFoundException, SQLException { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final CellSet cellSet = olapConnection.createStatement().executeOlapQuery( "select {[Measures].[Org Salary]} on 0,\n" + " Head([Employees].Members, 10) DIMENSION PROPERTIES DEPTH ON 1\n" + "from [HR]"); final CellSetAxis rowsAxis = cellSet.getAxes().get(1); assertEquals(10, rowsAxis.getPositionCount()); Member member0 = rowsAxis.getPositions().get(0).getMembers().get(0); assertEquals("All Employees", member0.getName()); assertEquals(0, member0.getDepth()); Member member1 = rowsAxis.getPositions().get(1).getMembers().get(0); assertEquals("[Employees].[Sheri Nowmer]", member1.getUniqueName()); assertEquals(1, member1.getDepth()); assertEquals(1, member1.getLevel().getDepth()); assertEquals( member0.getUniqueName(), member1.getParentMember().getUniqueName()); assertEquals(member0, member1.getParentMember()); Member member2 = rowsAxis.getPositions().get(2).getMembers().get(0); assertTrue( member2.getUniqueName().equals( "[Employees].[Derrick Whelply]") || member2.getUniqueName().equals( "[Employees].[Sheri Nowmer].[Derrick Whelply]")); assertEquals(2, member2.getDepth()); assertEquals(1, member2.getLevel().getDepth()); final Member parent = member2.getParentMember(); assertNotNull(parent); assertEquals("[Employees].[Sheri Nowmer]", parent.getUniqueName()); assertEquals(1, parent.getDepth()); assertEquals(member2.getLevel(), parent.getLevel()); assertEquals(member1, parent); final CellSetAxis filterAxis = cellSet.getFilterAxis(); assertEquals(1, filterAxis.getPositionCount()); final List<Position> positions = filterAxis.getPositions(); assertEquals(1, positions.size()); assertEquals(0, positions.get(0).getOrdinal()); // No WHERE clause, therefore slicer is empty (but not null). assertEquals(0, positions.get(0).getMembers().size()); } /** * Tests the type-derivation for * {@link org.olap4j.mdx.SelectNode#getFrom()} and the {@link CubeType} * class. * * @throws Throwable on error */ public void testCubeType() throws Throwable { if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() == TestContext.Tester.Flavor.REMOTE_XMLA) { // This test requires validator support. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final MdxParserFactory parserFactory = olapConnection.getParserFactory(); MdxParser mdxParser = parserFactory.createMdxParser(olapConnection); MdxValidator mdxValidator = parserFactory.createMdxValidator(olapConnection); SelectNode select = mdxParser.parseSelect( "select {[Gender]} on columns from [sales]\n" + "where [Time].[1997].[Q4]"); // CubeType // Before validation, we cannot ask for type try { final ParseTreeNode from = select.getFrom(); assertTrue(from instanceof IdentifierNode); Type type = from.getType(); fail("expected error, got " + type); } catch (UnsupportedOperationException e) { // ignore } select = mdxValidator.validateSelect(select); CubeType cubeType = (CubeType) select.getFrom().getType(); assertEquals("Sales", cubeType.getCube().getName()); assertNull(cubeType.getDimension()); assertNull(cubeType.getHierarchy()); assertNull(cubeType.getLevel()); // Different query based on same cube should have equal CubeType select = mdxParser.parseSelect( "select from [sales]"); select = mdxValidator.validateSelect(select); assertEquals(cubeType, select.getFrom().getType()); // Different query based on different cube should have different // CubeType select = mdxParser.parseSelect( "select from [warehouse and sales]"); select = mdxValidator.validateSelect(select); assertNotSame(cubeType, select.getFrom().getType()); } /** * Tests the type-derivation for query axes * ({@link org.olap4j.mdx.SelectNode#getAxisList()} and * {@link org.olap4j.mdx.SelectNode#getFilterAxis()}), and the * {@link org.olap4j.type.SetType}, * {@link org.olap4j.type.TupleType}, * {@link org.olap4j.type.MemberType} type subclasses. * * @throws Throwable on error */ public void testAxisType() throws Throwable { if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() == TestContext.Tester.Flavor.REMOTE_XMLA) { // This test requires validator support. return; } Class.forName(tester.getDriverClassName()); // connect using properties and no username/password connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final MdxParserFactory parserFactory = olapConnection.getParserFactory(); MdxParser mdxParser = parserFactory.createMdxParser(olapConnection); MdxValidator mdxValidator = parserFactory.createMdxValidator(olapConnection); SelectNode select = mdxParser.parseSelect( "select ([Gender], [Store]) on columns\n," + "{[Customers].[City].Members} on rows\n" + "from [sales]\n" + "where ([Time].[1997].[Q4], [Marital Status].[S])"); select = mdxValidator.validateSelect(select); // a query is not an expression, so does not have a type assertNull(select.getType()); final AxisNode columnsAxis = select.getAxisList().get(0); // an axis is not an expression, so does not have a type assertNull(columnsAxis.getType()); // ~ SetType final SetType setType = (SetType) columnsAxis.getExpression().getType(); assertNull(setType.getDimension()); assertNull(setType.getHierarchy()); assertNull(setType.getLevel()); assertNotNull(setType.toString()); final Type elementType = setType.getElementType(); // ~ TupleType assertTrue(elementType instanceof TupleType); TupleType tupleType = (TupleType) elementType; assertNotNull(tupleType.toString()); assertNull(tupleType.getDimension()); assertNull(tupleType.getHierarchy()); assertNull(tupleType.getLevel()); final Cube cube = ((CubeType) select.getFrom().getType()).getCube(); final Dimension storeDimension = cube.getDimensions().get("Store"); final Dimension genderDimension = cube.getDimensions().get("Gender"); final Dimension measuresDimension = cube.getDimensions().get("Measures"); final Dimension customersDimension = cube.getDimensions().get("Customers"); assertTrue(tupleType.usesDimension(storeDimension, false)); assertTrue(tupleType.usesDimension(genderDimension, false)); assertFalse(tupleType.usesDimension(measuresDimension, false)); // Other axis is a set of members // ~ MemberType final AxisNode rowsAxis = select.getAxisList().get(1); final Type rowsType = rowsAxis.getExpression().getType(); assertTrue(rowsType instanceof SetType); MemberType memberType = (MemberType) ((SetType) rowsType).getElementType(); assertNotNull(memberType.toString()); // MemberType.getMember is null because we know it belongs to the City // level, but no particular member of that level. assertNull("Customers", memberType.getMember()); assertEquals("City", memberType.getLevel().getName()); assertEquals("Customers", memberType.getHierarchy().getName()); assertEquals("Customers", memberType.getDimension().getName()); assertFalse(memberType.usesDimension(storeDimension, false)); assertTrue(memberType.usesDimension(customersDimension, false)); assertTrue(memberType.usesDimension(customersDimension, true)); // Filter final AxisNode filterAxis = select.getFilterAxis(); assertNull(filterAxis.getType()); final Type filterType = filterAxis.getExpression().getType(); assertTrue(filterType instanceof TupleType); assertEquals( "TupleType<MemberType<member=[Time].[Time].[1997].[Q4]>, MemberType<member=[Marital Status].[Marital Status].[S]>>", filterType.toString()); } public void testParseQueryWithNoFilter() throws Exception { if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() != Tester.Flavor.REMOTE_XMLA) { // This test requires validator support. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final MdxParserFactory parserFactory = olapConnection.getParserFactory(); MdxParser mdxParser = parserFactory.createMdxParser(olapConnection); MdxValidator mdxValidator = parserFactory.createMdxValidator(olapConnection); SelectNode select = mdxParser.parseSelect( "select ([Gender], [Store]) on columns\n," + "{[Customers].[City].Members} on rows\n" + "from [sales]"); select = mdxValidator.validateSelect(select); AxisNode filterAxis = select.getFilterAxis(); assertNotNull(filterAxis); assertNull(filterAxis.getExpression()); try { select = mdxParser.parseSelect( "select ([Gender], [Store]) on columns\n," + "{[Customers].[City].Members} on rows\n" + "from [sales]\n" + "where ()"); fail("expected parse error, got " + select); } catch (RuntimeException e) { assertTrue( TestContext.getStackTrace(e).indexOf( "Syntax error at [4:10], token ')'") >= 0); } } public void testValidateError() throws Exception { if (tester.getFlavor() == TestContext.Tester.Flavor.XMLA || tester.getFlavor() != Tester.Flavor.XMLA) { // This test requires validator support. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final MdxParserFactory parserFactory = olapConnection.getParserFactory(); MdxParser mdxParser = parserFactory.createMdxParser(olapConnection); MdxValidator mdxValidator = parserFactory.createMdxValidator(olapConnection); SelectNode select = mdxParser.parseSelect( "select ([Gender], [Store]) on columns\n," + "crossjoin([Customers].[City].Members, [Gender].members) on rows\n" + "from [sales]"); AxisNode filterAxis = select.getFilterAxis(); assertNotNull(filterAxis); assertNull(filterAxis.getExpression()); try { select = mdxValidator.validateSelect(select); fail("expected parse error, got " + select); } catch (OlapException e) { assertEquals("Validation error", e.getMessage()); final String stackTrace = TestContext.getStackTrace(e); assertTrue( stackTrace, stackTrace.contains( "Hierarchy '[Gender]' appears in more than one " + "independent axis.")); } } // TODO: test for HierarchyType // TODO: test for DimensionType // TODO: test for LevelType /** * Converts a list of members to a string, one per line. */ static String memberListToString(List<Member> list) { final StringBuilder buf = new StringBuilder(); for (Member member : list) { buf.append(member.getUniqueName()).append(TestContext.NL); } return buf.toString(); } public void testStatementCancel() throws Throwable { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final OlapStatement olapStatement = olapConnection.createStatement(); final Throwable[] exceptions = {null}; new Thread( new Runnable() { public void run() { try { Thread.sleep(1000); olapStatement.cancel(); } catch (Throwable e) { exceptions[0] = e; } } } ).start(); try { // Because XMLA doesn't pass cancel statements, // we set a query timeout to cleanup in the background if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { //Not existent in olap4ld //MondrianProperties.instance().QueryTimeout.set(5); } final CellSet cellSet = olapStatement.executeOlapQuery( "SELECT Filter(\n" + " [Product].Members *\n" + " [Customers].Members *\n" + " [Time].[Time].Members,\n" + " 1 = 0) on columns\n" + "from [Sales]"); fail( "expected exception indicating stmt had been canceled," + " got cellSet " + cellSet); } catch (OlapException e) { assertTrue( e.getMessage(), e.getMessage().indexOf("Query canceled") >= 0); } finally { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { //Not existent in olap4ld //MondrianProperties.instance().QueryTimeout.set(0); } } if (exceptions[0] != null) { throw exceptions[0]; } } public void testStatementTimeout() throws Throwable { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final OlapStatement olapStatement = olapConnection.createStatement(); try { olapStatement.setQueryTimeout(-1); fail("expected exception"); } catch (SQLException e) { assertTrue(e.getMessage().indexOf("illegal timeout value ") >= 0); } olapStatement.setQueryTimeout(1); if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { //Not existent in olap4ld //MondrianProperties.instance().QueryTimeout.set(1); } try { final CellSet cellSet = olapStatement.executeOlapQuery( "SELECT Filter(\n" + " [Store].Members * \n" + " [Customers].Members * \n" + " [Time].[Time].Members, 1 = 0) on columns\n" + "from [Sales]"); fail( "expected exception indicating timeout," + " got cellSet " + cellSet); } catch (OlapException e) { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { //Not existent in olap4ld //MondrianProperties.instance().QueryTimeout.set(0); } assertTrue( e.getMessage(), e.getMessage().indexOf("Query timeout of ") >= 0); } } public void testCellSetBug() throws SQLException { if (!Bug.BugOlap4j3126853Fixed) { return; } connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final OlapStatement olapStatement = olapConnection.createStatement(); // Note: substitute [Sales Ragged] for [Sales] below and the query // takes a very long time against mondrian's XMLA driver, because // mondrian has a performance bug assigning ordinals to ragged // hierarchies, and XMLA requests ask for member ordinals along with // the other attributes of members. CellSet cellSet = olapStatement.executeOlapQuery( "SELECT " + "{[Product].[All Products].[Drink].[Alcoholic Beverages].Children, [Product].[All Products].[Food].[Baked Goods].Children} ON COLUMNS, " + "CrossJoin([Store].[USA].[CA].Children, [Time].[1997].[Q1].Children) ON ROWS " + "FROM [Sales]"); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{}\n" + "Axis #1:\n" + "{[Product].[Drink].[Alcoholic Beverages].[Beer and Wine]}\n" + "{[Product].[Food].[Baked Goods].[Bread]}\n" + "Axis #2:\n" + "{[Store].[USA].[CA].[Alameda], [Time].[1997].[Q1].[1]}\n" + "{[Store].[USA].[CA].[Alameda], [Time].[1997].[Q1].[2]}\n" + "{[Store].[USA].[CA].[Alameda], [Time].[1997].[Q1].[3]}\n" + "{[Store].[USA].[CA].[Beverly Hills], [Time].[1997].[Q1].[1]}\n" + "{[Store].[USA].[CA].[Beverly Hills], [Time].[1997].[Q1].[2]}\n" + "{[Store].[USA].[CA].[Beverly Hills], [Time].[1997].[Q1].[3]}\n" + "{[Store].[USA].[CA].[Los Angeles], [Time].[1997].[Q1].[1]}\n" + "{[Store].[USA].[CA].[Los Angeles], [Time].[1997].[Q1].[2]}\n" + "{[Store].[USA].[CA].[Los Angeles], [Time].[1997].[Q1].[3]}\n" + "{[Store].[USA].[CA].[San Diego], [Time].[1997].[Q1].[1]}\n" + "{[Store].[USA].[CA].[San Diego], [Time].[1997].[Q1].[2]}\n" + "{[Store].[USA].[CA].[San Diego], [Time].[1997].[Q1].[3]}\n" + "{[Store].[USA].[CA].[San Francisco], [Time].[1997].[Q1].[1]}\n" + "{[Store].[USA].[CA].[San Francisco], [Time].[1997].[Q1].[2]}\n" + "{[Store].[USA].[CA].[San Francisco], [Time].[1997].[Q1].[3]}\n" + "Row #0: \n" + "Row #0: \n" + "Row #1: \n" + "Row #1: \n" + "Row #2: \n" + "Row #2: \n" + "Row #3: 22\n" + "Row #3: 63\n" + "Row #4: 28\n" + "Row #4: 59\n" + "Row #5: 28\n" + "Row #5: 39\n" + "Row #6: 70\n" + "Row #6: 51\n" + "Row #7: 89\n" + "Row #7: 51\n" + "Row #8: 27\n" + "Row #8: 54\n" + "Row #9: 54\n" + "Row #9: 51\n" + "Row #10: 38\n" + "Row #10: 48\n" + "Row #11: 64\n" + "Row #11: 55\n" + "Row #12: 6\n" + "Row #12: 2\n" + "Row #13: 3\n" + "Row #13: 7\n" + "Row #14: 2\n" + "Row #14: 10\n", TestContext.toString(cellSet)); } public void testCellSetWithCalcMember() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); final OlapStatement olapStatement = olapConnection.createStatement(); CellSet cellSet = olapStatement.executeOlapQuery( "WITH MEMBER [Measures].[Average Profit] AS" + "'[Measures].[Profit] / [Measures].[Sales Count]'" + "SELECT {[Measures].[Average Profit]} ON 0,\n" + "{[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].Children} ON 1\n" + "FROM [Sales] " + "WHERE ([Time.Weekly].[Week].[5])"); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{[Time].[Weekly].[1997].[5]}\n" + "Axis #1:\n" + "{[Measures].[Average Profit]}\n" + "Axis #2:\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Good]}\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Pearl]}\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Portsmouth]}\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Top Measure]}\n" + "{[Product].[Product].[Drink].[Alcoholic Beverages].[Beer and Wine].[Beer].[Walrus]}\n" + "Row #0: $2.17\n" + "Row #1: $2.60\n" + "Row #2: $6.03\n" + "Row #3: $1.54\n" + "Row #4: $5.11\n", TestContext.toString(cellSet)); } public void testBuildQuery() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); buildQuery(olapConnection, true); buildQuery(olapConnection, false); } private void buildQuery( OlapConnection olapConnection, boolean useCubeObject) throws OlapException { Catalog catalog = olapConnection.getOlapCatalogs() .get("FoodMart"); Schema schema = catalog.getSchemas().get("FoodMart"); Cube cube = schema.getCubes().get("Sales"); SelectNode query = new SelectNode(); ParseTreeNode cubeNode; if (useCubeObject) { cubeNode = IdentifierNode.parseIdentifier(cube.getUniqueName()); } else { cubeNode = new CubeNode(null, cube); } query.setFrom(cubeNode); AxisNode columnAxis = new AxisNode( null, false, Axis.COLUMNS, null, new CallNode( null, "MEMBERS", Syntax.Property, IdentifierNode.parseIdentifier("[Gender]"))); AxisNode rowAxis = new AxisNode( null, false, Axis.ROWS, null, new CallNode( null, "CHILDREN", Syntax.Property, IdentifierNode.parseIdentifier("[Customers].[USA]"))); query.getAxisList().add(columnAxis); query.getAxisList().add(rowAxis); OlapStatement statement = olapConnection.createStatement(); CellSet cellSet = statement.executeOlapQuery(query); TestContext.assertEqualsVerbose( "Axis #0:\n" + "{}\n" + "Axis #1:\n" + "{[Gender].[Gender].[All Gender]}\n" + "{[Gender].[Gender].[F]}\n" + "{[Gender].[Gender].[M]}\n" + "Axis #2:\n" + "{[Customers].[Customers].[USA].[CA]}\n" + "{[Customers].[Customers].[USA].[OR]}\n" + "{[Customers].[Customers].[USA].[WA]}\n" + "Row #0: 74,748\n" + "Row #0: 36,759\n" + "Row #0: 37,989\n" + "Row #1: 67,659\n" + "Row #1: 33,036\n" + "Row #1: 34,623\n" + "Row #2: 124,366\n" + "Row #2: 61,763\n" + "Row #2: 62,603\n", TestContext.toString(cellSet)); } public void testBuildQuery2() throws ClassNotFoundException, SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Schema schema = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart"); Cube cube = schema.getCubes().get("Sales"); Measure measure = cube.getMeasures().get(0); assertEquals("Unit Sales", measure.getName()); Dimension dimPromotionMedia = cube.getDimensions().get("Promotion Media"); // // IdentifierNode cubeNode = // new IdentifierNode( // new IdentifierNode.NameSegment(cube.getUniqueName())); CubeNode cubeNode = new CubeNode(null, cube); MemberNode measuresQuantity = new MemberNode(null, measure); HierarchyNode promotionHierarchyNode = new HierarchyNode(null, dimPromotionMedia.getDefaultHierarchy()); CallNode promotionChildren = new CallNode( null, "children", Syntax.Property, promotionHierarchyNode); // List<IdentifierNode> columnDimensionProperties = new ArrayList<IdentifierNode>(); AxisNode columnAxis = new AxisNode( null, false, Axis.COLUMNS, columnDimensionProperties, new CallNode(null, "{}", Syntax.Braces, measuresQuantity)); List<IdentifierNode> rowDimensionProperties = new ArrayList<IdentifierNode>(); AxisNode rowAxis = new AxisNode( null, false, Axis.ROWS, rowDimensionProperties, new CallNode(null, "{}", Syntax.Braces, promotionChildren)); // SelectNode query = new SelectNode(); query.setFrom(cubeNode); query.getAxisList().add(columnAxis); query.getAxisList().add(rowAxis); // OlapStatement statement = olapConnection.createStatement(); CellSet cellSet = statement.executeOlapQuery(query); final StringWriter sw = new StringWriter(); final PrintWriter pw = new PrintWriter(sw); for (Position row : cellSet.getAxes().get(1)) { for (Position column : cellSet.getAxes().get(0)) { pw.print("ROW:"); for (Member member : row.getMembers()) { pw.print("[" + member.getName() + "]"); } pw.print(" COL:"); for (Member member : column.getMembers()) { pw.print("[" + member.getName() + "]"); } Cell cell = cellSet.getCell(column, row); pw.println(" CELL:" + cell.getFormattedValue()); } } pw.flush(); TestContext.assertEqualsVerbose( "ROW:[Bulk Mail] COL:[Unit Sales] CELL:4,320\n" + "ROW:[Cash Register Handout] COL:[Unit Sales] CELL:6,697\n" + "ROW:[Daily Paper] COL:[Unit Sales] CELL:7,738\n" + "ROW:[Daily Paper, Radio] COL:[Unit Sales] CELL:6,891\n" + "ROW:[Daily Paper, Radio, TV] COL:[Unit Sales] CELL:9,513\n" + "ROW:[In-Store Coupon] COL:[Unit Sales] CELL:3,798\n" + "ROW:[No Media] COL:[Unit Sales] CELL:195,448\n" + "ROW:[Product Attachment] COL:[Unit Sales] CELL:7,544\n" + "ROW:[Radio] COL:[Unit Sales] CELL:2,454\n" + "ROW:[Street Handout] COL:[Unit Sales] CELL:5,753\n" + "ROW:[Sunday Paper] COL:[Unit Sales] CELL:4,339\n" + "ROW:[Sunday Paper, Radio] COL:[Unit Sales] CELL:5,945\n" + "ROW:[Sunday Paper, Radio, TV] COL:[Unit Sales] CELL:2,726\n" + "ROW:[TV] COL:[Unit Sales] CELL:3,607\n", sw.toString()); } /** * Verifies the order of dimensions; they must conform to their * ordinal value. * @throws Exception If something turns sour. */ public void testCubeDimensionsOrder() throws Exception { String dimNames = "[Measures];[Store];[Store Size in SQFT];" + "[Store Type];[Time];[Product];[Promotion Media];[Promotions];" + "[Customers];[Education Level];[Gender];[Marital Status];" + "[Yearly Income];"; Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Cube cube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Sales"); StringBuilder sb = new StringBuilder(); for (Dimension dimension : cube.getDimensions()) { sb.append(dimension.getUniqueName()) .append(";"); } TestContext.assertEqualsVerbose( dimNames, sb.toString()); } public void testCubesDrillthrough() throws Exception { Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); Cube cube = olapConnection .getOlapCatalogs() .get("FoodMart") .getSchemas() .get("FoodMart") .getCubes() .get("Sales"); switch (tester.getFlavor()) { case MONDRIAN: assertTrue(cube.isDrillThroughEnabled()); break; case REMOTE_XMLA : case XMLA: assertFalse(cube.isDrillThroughEnabled()); break; default: fail(); break; } } /** * This is a test case for * <a href="http://jira.pentaho.com/browse/MONDRIAN-893"> * MONDRIAN-893</a>. * * <p>Test two arguments as part of the return clause, * on a constrained cell. One argument is a level * which returns a numeric key, the other is a level * as a numeric key. */ public void testCubesDrillthroughReturnClause() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Customers].[Name], [Gender].[Gender]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:5956.0,M,", "ROW:6013.0,M,", "ROW:7293.0,M,", "ROW:7683.0,F,", "ROW:7683.0,F," }); } finally { rs.close(); } } /** * Same test as {@link #testCubesDrillthroughReturnClause()} * but with a single column returned. */ public void testCubesDrillthroughReturnClause2() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Customers].[Name]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:5956.0,", "ROW:6013.0,", "ROW:7293.0,", "ROW:7683.0,", "ROW:7683.0," }); } finally { rs.close(); } } /** * Same test as {@link #testCubesDrillthroughReturnClause()} * but without selecting anything except the slicer axis. */ public void testCubesDrillthroughReturnClause3() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Customers].[Name]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:5956.0,", "ROW:6013.0,", "ROW:7293.0,", "ROW:7683.0,", "ROW:7683.0," }); } finally { rs.close(); } } /** * A drillthrough on a cell which is part of a query which doesn't * include any measure on its axis, but has 2 in the RETURN clause. */ public void testCubesDrillthroughReturnClause4() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Promotions].[One Day Sale]} on columns,\n" + "{[Store].[Store City].[Walla Walla]} on rows\n" + "from [Sales]\n" + "where ([Product].[Product Category].[Pizza])\n" + "RETURN [Measures].[Unit Sales], [Measures].[Store Sales]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:1.0,0.8," }); } finally { rs.close(); } } /** * A drillthrough on a cell which is part of a query which doesn't * include any measure on its axis, but has 1 in the RETURN clause. */ public void testCubesDrillthroughReturnClause5() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Promotions].[One Day Sale]} on columns,\n" + "{[Store].[Store City].[Walla Walla]} on rows\n" + "from [Sales]\n" + "where ([Product].[Product Category].[Pizza])\n" + "RETURN [Measures].[Store Sales]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:0.8," }); } finally { rs.close(); } } /** * Drillthrough with a measure on the axis, but returning * a different measure as part of the RETURN clause. */ public void testCubesDrillthroughReturnClause6() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Measures].[Store Sales]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:12.34," }); } finally { rs.close(); } } /** * Drillthrough with a measure on the axis and returning * a dimension and a SQL generated measure as part of the RETURN clause. */ public void testCubesDrillthroughReturnClause7() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Customers], [Measures].[Promotion Sales]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:USA,12.34," }); } finally { rs.close(); } } /** * Drillthrough with a measure on the axis and returning * a level and a SQL generated measure as part of the RETURN clause. */ public void testCubesDrillthroughReturnClause8() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])\n" + "RETURN [Customers].[City], [Measures].[Promotion Sales]"); try { assertDrillRowsEquals( rs, new String[] { "ROW:WA,Walla Walla,12.34," }); } finally { rs.close(); } } /** * Drillthrough with a measure on the axis and no RETURN * clause specified. By default we return whatever is * in the slicer, plus the individual measure values. */ public void testCubesDrillthroughReturnClause9() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH\n" + "SELECT {[Measures].[Unit Sales]} on columns\n" + "from [Sales]\n" + "where ([Promotions].[One Day Sale],\n" + " [Store].[Store City].[Walla Walla],\n" + " [Product].[Product Category].[Bread])"); try { assertDrillRowsEquals( rs, new String[] { "ROW:WA,Walla Walla,Store 22,1997.0,Q3,8.0,Food,Baked Goods,Bread,Muffins,Great,Great Muffins,One Day Sale,1.0,", "ROW:WA,Walla Walla,Store 22,1997.0,Q3,8.0,Food,Baked Goods,Bread,Muffins,Modell,Modell Cranberry Muffins,One Day Sale,1.0,", "ROW:WA,Walla Walla,Store 22,1997.0,Q3,8.0,Food,Baked Goods,Bread,Sliced Bread,Colony,Colony White Bread,One Day Sale,2.0,", "ROW:WA,Walla Walla,Store 22,1997.0,Q3,9.0,Food,Baked Goods,Bread,Muffins,Modell,Modell Blueberry Muffins,One Day Sale,1.0,", "ROW:WA,Walla Walla,Store 22,1997.0,Q3,9.0,Food,Baked Goods,Bread,Sliced Bread,Sphinx,Sphinx Wheat Bread,One Day Sale,1.0," }); } finally { rs.close(); } } /** * Drillthrough with a measure on the axis, with * two different measures as part of the RETURN clause. */ public void testCubesDrillthroughReturnClause10() throws Exception { if (tester.getFlavor().equals(Flavor.XMLA) || tester.getFlavor().equals(Flavor.REMOTE_XMLA)) { // TODO: Drillthrough is not implemented in XMLA yet. return; } Class.forName(tester.getDriverClassName()); connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); ResultSet rs = olapConnection.createStatement().executeQuery( "DRILLTHROUGH SELECT from [Sales] where ([Promotions].[One Day Sale], [Store].[Store City].[Walla Walla], [Product].[Product Category].[Bread]) RETURN [Time].[Day], [Measures].[Store Sales], [Measures].[Store Cost]"); assertDrillRowsEquals( rs, new String[] { "ROW:1997.0,34.0,14.0,9.12,3.443,", "ROW:1997.0,38.0,10.0,3.22,1.288," }); } private void assertDrillRowsEquals( ResultSet rs, String[] expected) throws Exception { final List<String> rows = new ArrayList<String>(); StringBuilder sb = new StringBuilder(); while (rs.next()) { sb.setLength(0); sb.append("ROW:"); for (int i = 0; i < rs.getMetaData().getColumnCount(); i++) { sb.append( normalizeTrailingZeros( rs.getObject(i + 1))); sb.append(","); } rows.add(sb.toString()); } Collections.sort(rows); Arrays.sort(expected); for (int i = 0; i < rows.size(); i++) { assertEquals( expected[i], rows.get(i)); } } /** * This method normalizes the number of trailing zeros across * the different FoodMart databases / test environments. All numbers * are returned with format: '#0.0#' */ private String normalizeTrailingZeros(Object str) { if (str == null) { return null; } final Double number; try { number = Double.parseDouble(String.valueOf(str)); } catch (NumberFormatException e) { return String.valueOf(str); } return String.valueOf(number); } /** * Query with dimension properties. * * <p>Test case for * <a href="http://sourceforge.net/tracker/?func=detail&aid=2951656&group_id=168953&atid=848534"> * bug 2951656, "Olap4j should not throw IllegalArgumentException"</a>. * * @throws java.sql.SQLException on error */ public void _testDimensionProperties() throws SQLException { connection = tester.createConnection(); OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); String mdx = "select {[Product].[Product Family].Members} ON COLUMNS,\n" + "{[Store].[Store Name].Members}\n" + " DIMENSION PROPERTIES [Store].[Store Name].[Store Sqft] ON ROWS\n" + "from [Sales]\n" + "where [Measures].[Unit Sales]"; PreparedOlapStatement pstmt = olapConnection.prepareOlapStatement(mdx); pstmt.executeQuery(); } /** * <p>This test relates to * sourceforge.net/projects/olap4j/forums/forum/577988/topic/3803726 * * <p>Drivers should return a transaction isolation of * {@link Connection#TRANSACTION_NONE} */ public void testTransactionIsolation() throws SQLException { connection = tester.createConnection(); assertEquals( Connection.TRANSACTION_NONE, connection.getTransactionIsolation()); } public void testSchemaAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final List<String> list = new ArrayList<String>(); for (Schema schema : olapConnection.getOlapSchemas()) { list.add(schema.getName()); } assertListsEquals( Arrays.asList( "FoodMart"), list); } finally { olapConnection.setRoleName(null); } } public void testCubeAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final List<String> list = new ArrayList<String>(); for (Cube cube : olapConnection.getOlapSchema().getCubes()) { list.add(cube.getName()); } assertListsEquals( Arrays.asList( "Sales"), list); olapConnection.setRoleName("No HR Cube"); list.clear(); for (Cube cube : olapConnection.getOlapSchema().getCubes()) { list.add(cube.getName()); } assertListsEquals( Arrays.asList( "Sales Ragged", "Warehouse", "Warehouse and Sales", "Sales 2", "Store", "Sales"), list); } finally { olapConnection.setRoleName(null); } } public void testDimensionAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final List<String> list = new ArrayList<String>(); for (Dimension dimension : olapConnection.getOlapSchema() .getCubes().get("Sales").getDimensions()) { list.add(dimension.getName()); } assertListsEquals( Arrays.asList( "Measures", "Store", "Store Size in SQFT", "Store Type", "Time", "Product", "Promotion Media", "Promotions", "Customers", "Education Level", "Marital Status", "Yearly Income"), list); } finally { olapConnection.setRoleName(null); } } public void testLevelAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final List<String> list = new ArrayList<String>(); for (Level level : olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Customers") .getHierarchies().get(0) .getLevels()) { list.add(level.getName()); } assertListsEquals( Arrays.asList( "State Province", "City"), list); } finally { olapConnection.setRoleName(null); } } public void testLevelMembersAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final List<String> list = new ArrayList<String>(); for (Member member : olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Store") .getHierarchies().get(0) .getLevels().get(0) .getMembers()) { list.add(member.getName()); } assertEquals( Arrays.asList( "USA"), list); list.clear(); for (Member member : olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Store") .getHierarchies().get(0) .getLevels().get(1) .getMembers()) { list.add(member.getName()); } assertListsEquals( Arrays.asList( "CA"), list); list.clear(); for (Member member : olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Store") .getHierarchies().get(0) .getLevels().get(2) .getMembers()) { list.add(member.getName()); } assertListsEquals( Arrays.asList( "Alameda", "Beverly Hills", "San Diego", "San Francisco"), list); } finally { olapConnection.setRoleName(null); } } public void testParentChildAccessControl() throws Exception { if (tester.getFlavor() == Flavor.XMLA || tester.getFlavor() == Flavor.REMOTE_XMLA) { // TODO Implement a way to pass an XmlaRequestCallback to // the tested servlet so we can pass the roles down. return; } connection = tester.createConnection(); final OlapConnection olapConnection = tester.getWrapper().unwrap(connection, OlapConnection.class); try { olapConnection.setRoleName("California manager"); final Member city = olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Customers") .getHierarchies().get(0) .getLevels().get("City") .getMembers().get(0); assertEquals( 0, city.getChildMembers().size()); final Member state = olapConnection.getOlapSchema() .getCubes().get("Sales") .getDimensions().get("Customers") .getHierarchies().get(0) .getLevels().get("State Province") .getMembers().get(0); assertNull(state.getParentMember()); } finally { olapConnection.setRoleName(null); } } protected void assertListsEquals(List<String> list1, List<String> list2) { Collections.sort(list1); Collections.sort(list2); assertEquals(list1, list2); } } // End ConnectionTest.java