All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hsqldb.test.TestGroupByHaving Maven / Gradle / Ivy

There is a newer version: 2.7.2
Show newest version
/* Copyright (c) 2001-2019, The HSQL Development Group
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * Neither the name of the HSQL Development Group nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */


package org.hsqldb.test;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

import junit.framework.TestCase;

/**
 * Test cases for HSQL aggregates and HAVING clause.
 *
 * @author Tony Lai ((tony_lai@users dot sourceforge.net)
 */

// fredt@users - modified to remove dependecy on DBUnit
public class TestGroupByHaving extends TestCase {

    //------------------------------------------------------------
    // Class variables
    //------------------------------------------------------------
    private static final String databaseDriver   = "org.hsqldb.jdbc.JDBCDriver";
    private static final String databaseURL      = "jdbc:hsqldb:mem:.";
    private static final String databaseUser     = "sa";
    private static final String databasePassword = "";

    //------------------------------------------------------------
    // Instance variables
    //------------------------------------------------------------
    private Connection conn;
    private Statement  stmt;

    //------------------------------------------------------------
    // Constructors
    //------------------------------------------------------------

    /**
     * Constructs a new SubselectTest.
     */
    public TestGroupByHaving(String s) {
        super(s);
    }

    //------------------------------------------------------------
    // Class methods
    //------------------------------------------------------------
    protected static Connection getJDBCConnection() throws SQLException {
        return DriverManager.getConnection(databaseURL, databaseUser,
                                           databasePassword);
    }

    protected void setUp() throws Exception {

        super.setUp();

        if (conn != null) {
            return;
        }

        Class.forName(databaseDriver);

        conn = getJDBCConnection();
        stmt = conn.createStatement();

        // I decided not the use the "IF EXISTS" clause since it is not a
        // SQL standard.
        try {

//            stmt.execute("drop table employee");
            stmt.execute("drop table employee if exists");
        } catch (Exception x) {}

        stmt.execute("create table employee(id int, "
                     + "firstname VARCHAR(50), " + "lastname VARCHAR(50), "
                     + "salary decimal(10, 2), " + "superior_id int, "
                     + "CONSTRAINT PK_employee PRIMARY KEY (id), "
                     + "CONSTRAINT FK_superior FOREIGN KEY (superior_id) "
                     + "REFERENCES employee(ID))");
        addEmployee(1, "Mike", "Smith", 160000, -1);
        addEmployee(2, "Mary", "Smith", 140000, -1);

        // Employee under Mike
        addEmployee(10, "Joe", "Divis", 50000, 1);
        addEmployee(11, "Peter", "Mason", 45000, 1);
        addEmployee(12, "Steve", "Johnson", 40000, 1);
        addEmployee(13, "Jim", "Hood", 35000, 1);

        // Employee under Mike
        addEmployee(20, "Jennifer", "Divis", 60000, 2);
        addEmployee(21, "Helen", "Mason", 50000, 2);
        addEmployee(22, "Daisy", "Johnson", 40000, 2);
        addEmployee(23, "Barbara", "Hood", 30000, 2);
    }

    protected void tearDown() throws Exception {

        // I decided not the use the "IF EXISTS" clause since it is not a
        // SQL standard.
        try {

//            stmt.execute("drop table employee");
            stmt.execute("drop table employee if exists");
        } catch (Exception x) {}

        if (stmt != null) {
            stmt.close();

            stmt = null;
        }

        if (conn != null) {
            conn.close();

            conn = null;
        }

        super.tearDown();

    }

    private void addEmployee(int id, String firstName, String lastName,
                             double salary, int superiorId) throws Exception {

        stmt.execute("insert into employee values(" + id + ", '" + firstName
                     + "', '" + lastName + "', " + salary + ", "
                     + (superiorId <= 0 ? "null"
                                        : ("" + superiorId)) + ")");
    }

    /**
     * Tests aggregated selection with a GROUP_BY clause.  This is
     * a normal use of the GROUP_BY clause.  The first two employees
     * do not have a superior, and must be grouped within the same group,
     * according to GROUP_BY standard.
     */
    public void testAggregatedGroupBy() throws SQLException {

        String sql = "select avg(salary), max(id) from employee "
                     + "group by superior_id " + "order by superior_id " + "";
        Object[][] expected = new Object[][] {
            {
                Double.valueOf(150000), Integer.valueOf(2)
            }, {
                Double.valueOf(42500), Integer.valueOf(13)
            }, {
                Double.valueOf(45000), Integer.valueOf(23)
            },
        };

        compareResults(sql, expected, "00000");
    }

    /**
     * Tests aggregated selection with a GROUP_BY clause and a
     * HAVING clause.
     * 

* This is a typical use of the GROUP_BY + HAVING clause. * The first two employees are eliminated due to the HAVING * condition. *

* This test uses aggregated function to eliminate first group. */ public void testAggregatedGroupByHaving1() throws SQLException { String sql = "select avg(salary), max(id) from employee " + "group by superior_id " + "having max(id) > 5 " + "order by superior_id " + ""; Object[][] expected = new Object[][] { { Double.valueOf(42500), Integer.valueOf(13) }, { Double.valueOf(45000), Integer.valueOf(23) }, }; compareResults(sql, expected, "00000"); } /** * Tests aggregated selection with a GROUP_BY clause and a * HAVING clause. *

* This is a typical use of the GROUP_BY + HAVING clause. * The first two employees are eliminated due to the HAVING * condition. *

* This test uses GROUP_BY column to eliminate first group. */ public void testAggregatedGroupByHaving2() throws SQLException { String sql = "select avg(salary), max(id) from employee " + "group by superior_id " + "having superior_id is not null " + "order by superior_id " + ""; Object[][] expected = new Object[][] { { Double.valueOf(42500), Integer.valueOf(13) }, { Double.valueOf(45000), Integer.valueOf(23) }, }; compareResults(sql, expected, "00000"); } /** * Tests an unusual usage of the HAVING clause, without a * GROUP BY clause. *

* Only one row is returned by the aggregate selection without a * GROUP BY clause. The HAVING clause is applied to the * only returned row. In this case, the HAVING condition is * satisfied. */ public void testHavingWithoutGroupBy1() throws SQLException { String sql = "select avg(salary), max(id) from employee " + "having avg(salary) > 1000 " + ""; Object[][] expected = new Object[][] { { Double.valueOf(65000), Integer.valueOf(23) }, }; compareResults(sql, expected, "00000"); } /** * Tests an unusual usage of the HAVING clause, without a * GROUP BY clause. *

* Only one row is returned by the aggregate selection without a * GROUP BY clause. The HAVING clause is applied to the * only returned row. In this case, the HAVING condition is * NOT satisfied. */ public void testHavingWithoutGroupBy2() throws SQLException { String sql = "select avg(salary), max(id) from employee " + "having avg(salary) > 1000000 " + ""; Object[][] expected = new Object[][]{}; compareResults(sql, expected, "00000"); } /** * Tests an invalid HAVING clause that contains columns not in * the GROUP BY clause. A SQLException should be thrown. */ public void testInvalidHaving() throws SQLException { String sql = "select avg(salary), max(id) from employee " + "group by lastname " + "having (max(id) > 1) and (superior_id > 1) " + ""; Object[][] expected = new Object[][]{}; compareResults(sql, expected, "42573"); } //------------------------------------------------------------ // Helper methods //------------------------------------------------------------ private void compareResults(String sql, Object[][] rows, String sqlState) throws SQLException { ResultSet rs = null; try { rs = stmt.executeQuery(sql); assertTrue("Statement <" + sql + "> \nexpecting error code: " + sqlState, ("00000".equals(sqlState))); } catch (SQLException sqlx) { if (!sqlx.getSQLState().equals(sqlState)) { sqlx.printStackTrace(); } assertTrue("Statement <" + sql + "> \nthrows wrong error code: " + sqlx.getErrorCode() + " expecting error code: " + sqlState, (sqlx.getSQLState().equals(sqlState))); return; } int rowCount = 0; int colCount = rows.length > 0 ? rows[0].length : 0; while (rs.next()) { assertTrue("Statement <" + sql + "> \nreturned too many rows.", (rowCount < rows.length)); Object[] columns = rows[rowCount]; for (int col = 1, i = 0; i < colCount; i++, col++) { Object result = null; Object expected = columns[i]; if (expected == null) { result = rs.getString(col); result = rs.wasNull() ? null : result; } else if (expected instanceof String) { result = rs.getString(col); } else if (expected instanceof Double) { result = Double.valueOf(rs.getString(col)); } else if (expected instanceof Integer) { result = Integer.valueOf(rs.getInt(col)); } assertEquals("Statement <" + sql + "> \nreturned wrong value.", columns[i], result); } rowCount++; } assertEquals("Statement <" + sql + "> \nreturned wrong number of rows.", rows.length, rowCount); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy