/** * @file Statement_test.cpp * @ingroup tests * @brief Test of a SQLiteCpp Statement. * * Copyright (c) 2012-2016 Sebastien Rombauts (sebastien.rombauts@gmail.com) * * Distributed under the MIT License (MIT) (See accompanying file LICENSE.txt * or copy at http://opensource.org/licenses/MIT) */ #include #include #include #include TEST(Statement, invalid) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE|SQLITE_OPEN_CREATE); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); EXPECT_EQ(SQLITE_OK, db.getExtendedErrorCode()); // Compile a SQL query, but without any table in the database EXPECT_THROW(SQLite::Statement query(db, "SELECT * FROM test"), SQLite::Exception); EXPECT_EQ(SQLITE_ERROR, db.getErrorCode()); EXPECT_EQ(SQLITE_ERROR, db.getExtendedErrorCode()); EXPECT_EQ(0, db.exec("CREATE TABLE test (id INTEGER PRIMARY KEY, value TEXT)")); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); EXPECT_EQ(SQLITE_OK, db.getExtendedErrorCode()); // Compile a SQL query with no parameter SQLite::Statement query(db, "SELECT * FROM test"); EXPECT_STREQ("SELECT * FROM test", query.getQuery().c_str()); EXPECT_EQ(2, query.getColumnCount ()); EXPECT_FALSE(query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.isColumnNull(-1), SQLite::Exception); EXPECT_THROW(query.isColumnNull(0), SQLite::Exception); EXPECT_THROW(query.isColumnNull(1), SQLite::Exception); EXPECT_THROW(query.isColumnNull(2), SQLite::Exception); EXPECT_THROW(query.getColumn(-1), SQLite::Exception); EXPECT_THROW(query.getColumn(0), SQLite::Exception); EXPECT_THROW(query.getColumn(1), SQLite::Exception); EXPECT_THROW(query.getColumn(2), SQLite::Exception); query.reset(); EXPECT_FALSE(query.isOk()); EXPECT_FALSE(query.isDone()); query.executeStep(); EXPECT_FALSE(query.isOk()); EXPECT_TRUE( query.isDone()); query.reset(); EXPECT_FALSE(query.isOk()); EXPECT_FALSE(query.isDone()); query.reset(); EXPECT_THROW(query.bind(-1, 123), SQLite::Exception); EXPECT_THROW(query.bind(0, 123), SQLite::Exception); EXPECT_THROW(query.bind(1, 123), SQLite::Exception); EXPECT_THROW(query.bind(2, 123), SQLite::Exception); EXPECT_THROW(query.bind(0, "abc"), SQLite::Exception); EXPECT_THROW(query.bind(0), SQLite::Exception); EXPECT_EQ(SQLITE_RANGE, db.getErrorCode()); EXPECT_EQ(SQLITE_RANGE, db.getExtendedErrorCode()); query.exec(); // exec() instead of executeStep() as there is no result EXPECT_THROW(query.isColumnNull(0), SQLite::Exception); EXPECT_THROW(query.getColumn(0), SQLite::Exception); EXPECT_THROW(query.exec(), SQLite::Exception); // exec() shall throw as it needs to be reseted // Add a first row EXPECT_EQ(1, db.exec("INSERT INTO test VALUES (NULL, \"first\")")); EXPECT_EQ(1, db.getLastInsertRowid()); EXPECT_EQ(1, db.getTotalChanges()); query.reset(); EXPECT_FALSE(query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.exec(), SQLite::Exception); // exec() shall throw as it does not expect a result } TEST(Statement, executeStep) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); // Create a new table EXPECT_EQ(0, db.exec("CREATE TABLE test (id INTEGER PRIMARY KEY, msg TEXT, int INTEGER, double REAL)")); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); // Create a first row EXPECT_EQ(1, db.exec("INSERT INTO test VALUES (NULL, \"first\", 123, 0.123)")); EXPECT_EQ(1, db.getLastInsertRowid()); // Compile a SQL query SQLite::Statement query(db, "SELECT * FROM test"); EXPECT_STREQ("SELECT * FROM test", query.getQuery().c_str()); EXPECT_EQ(4, query.getColumnCount()); // Get the first row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); const sqlite3_int64 id = query.getColumn(0); const std::string msg = query.getColumn(1); const int integer = query.getColumn(2); const double real = query.getColumn(3); EXPECT_EQ(1, id); EXPECT_EQ("first", msg); EXPECT_EQ(123, integer); EXPECT_EQ(0.123, real); // Step one more time to discover there is nothing more query.executeStep(); EXPECT_FALSE(query.isOk()); EXPECT_TRUE (query.isDone()); // "done" is "the end" // Step after "the end" throw an exception EXPECT_THROW(query.executeStep(), SQLite::Exception); } TEST(Statement, bindings) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); // Create a new table EXPECT_EQ(0, db.exec("CREATE TABLE test (id INTEGER PRIMARY KEY, msg TEXT, int INTEGER, double REAL)")); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); // Insertion with binded values SQLite::Statement insert(db, "INSERT INTO test VALUES (NULL, ?, ?, ?)"); // First row with text/int/double insert.bind(1, "first"); insert.bind(2, 123); insert.bind(3, 0.123); EXPECT_EQ(1, insert.exec()); EXPECT_EQ(SQLITE_DONE, db.getErrorCode()); // Compile a SQL query to check the result SQLite::Statement query(db, "SELECT * FROM test"); EXPECT_STREQ("SELECT * FROM test", query.getQuery().c_str()); EXPECT_EQ(4, query.getColumnCount()); // Check the first row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_EQ (1, query.getColumn(0).getInt64()); EXPECT_STREQ("first", query.getColumn(1).getText()); EXPECT_EQ (123, query.getColumn(2).getInt()); EXPECT_EQ (0.123, query.getColumn(3).getDouble()); // reset() without clearbindings() insert.reset(); // Second row with the same exact values because clearbindings() was not called EXPECT_EQ(1, insert.exec()); EXPECT_EQ(SQLITE_DONE, db.getErrorCode()); // Check the second row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_EQ (2, query.getColumn(0).getInt64()); EXPECT_STREQ("first", query.getColumn(1).getText()); EXPECT_EQ (123, query.getColumn(2).getInt()); EXPECT_EQ (0.123, query.getColumn(3).getDouble()); // reset() with clearbindings() and no more bindings insert.reset(); insert.clearBindings(); // Third row with the all null values because clearbindings() was called EXPECT_EQ(1, insert.exec()); EXPECT_EQ(SQLITE_DONE, db.getErrorCode()); // Check the third row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_EQ (3, query.getColumn(0).getInt64()); EXPECT_TRUE (query.isColumnNull(1)); EXPECT_STREQ("", query.getColumn(1).getText()); EXPECT_TRUE (query.isColumnNull(2)); EXPECT_EQ (0, query.getColumn(2).getInt()); EXPECT_TRUE (query.isColumnNull(3)); EXPECT_EQ (0.0, query.getColumn(3).getDouble()); // reset() with clearbindings() and new bindings insert.reset(); insert.clearBindings(); // Fourth row with string/int64/float const std::string second("second"); const sqlite_int64 int64 = 12345678900000LL; const float fl32 = 0.123f; insert.bind(1, second); insert.bind(2, int64); insert.bind(3, fl32); EXPECT_EQ(1, insert.exec()); EXPECT_EQ(SQLITE_DONE, db.getErrorCode()); // Check the fourth row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_EQ(4, query.getColumn(0).getInt64()); EXPECT_EQ(second, query.getColumn(1).getText()); EXPECT_EQ(12345678900000LL, query.getColumn(2).getInt64()); EXPECT_EQ(0.123f, query.getColumn(3).getDouble()); } TEST(Statement, isColumnNull) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); ASSERT_EQ(SQLITE_OK, db.getErrorCode()); // Create a new table EXPECT_EQ(0, db.exec("CREATE TABLE test (msg TEXT, int INTEGER, double REAL)")); ASSERT_EQ(SQLITE_OK, db.getErrorCode()); // Create a first row with no null values, then other rows with each time a NULL value ASSERT_EQ(1, db.exec("INSERT INTO test VALUES (\"first\", 123, 0.123)")); ASSERT_EQ(1, db.exec("INSERT INTO test VALUES (NULL, 123, 0.123)")); ASSERT_EQ(1, db.exec("INSERT INTO test VALUES (\"first\", NULL, 0.123)")); ASSERT_EQ(1, db.exec("INSERT INTO test VALUES (\"first\", 123, NULL)")); // Compile a SQL query const std::string select("SELECT * FROM test"); SQLite::Statement query(db, select); EXPECT_EQ(select, query.getQuery()); EXPECT_EQ(3, query.getColumnCount()); // Get the first non-null row query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.isColumnNull(-1), SQLite::Exception); EXPECT_EQ(false, query.isColumnNull(0)); EXPECT_EQ(false, query.isColumnNull(1)); EXPECT_EQ(false, query.isColumnNull(2)); EXPECT_THROW(query.isColumnNull(3), SQLite::Exception); // Get the second row with null text query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.isColumnNull(-1), SQLite::Exception); EXPECT_EQ(true, query.isColumnNull(0)); EXPECT_EQ(false, query.isColumnNull(1)); EXPECT_EQ(false, query.isColumnNull(2)); EXPECT_THROW(query.isColumnNull(3), SQLite::Exception); // Get the second row with null integer query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.isColumnNull(-1), SQLite::Exception); EXPECT_EQ(false, query.isColumnNull(0)); EXPECT_EQ(true, query.isColumnNull(1)); EXPECT_EQ(false, query.isColumnNull(2)); EXPECT_THROW(query.isColumnNull(3), SQLite::Exception); // Get the third row with null float query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); EXPECT_THROW(query.isColumnNull(-1), SQLite::Exception); EXPECT_EQ(false, query.isColumnNull(0)); EXPECT_EQ(false, query.isColumnNull(1)); EXPECT_EQ(true, query.isColumnNull(2)); EXPECT_THROW(query.isColumnNull(3), SQLite::Exception); } TEST(Statement, getColumnByName) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); EXPECT_EQ(SQLITE_OK, db.getExtendedErrorCode()); // Create a new table EXPECT_EQ(0, db.exec("CREATE TABLE test (id INTEGER PRIMARY KEY, msg TEXT, int INTEGER, double REAL)")); EXPECT_EQ(SQLITE_OK, db.getErrorCode()); EXPECT_EQ(SQLITE_OK, db.getExtendedErrorCode()); // Create a first row EXPECT_EQ(1, db.exec("INSERT INTO test VALUES (NULL, \"first\", 123, 0.123)")); EXPECT_EQ(1, db.getLastInsertRowid()); EXPECT_EQ(1, db.getTotalChanges()); // Compile a SQL query SQLite::Statement query(db, "SELECT * FROM test"); EXPECT_STREQ("SELECT * FROM test", query.getQuery().c_str()); EXPECT_EQ(4, query.getColumnCount()); query.executeStep(); EXPECT_TRUE (query.isOk()); EXPECT_FALSE(query.isDone()); // Look for non-existing columns EXPECT_THROW(query.getColumn("unknown"), SQLite::Exception); EXPECT_THROW(query.getColumn(""), SQLite::Exception); const std::string msg = query.getColumn("msg"); const int integer = query.getColumn("int"); const double real = query.getColumn("double"); EXPECT_EQ("first", msg); EXPECT_EQ(123, integer); EXPECT_EQ(0.123, real); } TEST(Statement, getName) { // Create a new database SQLite::Database db(":memory:", SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); EXPECT_EQ(0, db.exec("CREATE TABLE test (id INTEGER PRIMARY KEY, msg TEXT)")); // Compile a SQL query, using the "id" column name as-is, but aliasing the "msg" column with new name "value" SQLite::Statement query(db, "SELECT id, msg as value FROM test"); query.executeStep(); const std::string name0 = query.getColumnName(0); const std::string name1 = query.getColumnName(1); EXPECT_EQ("id", name0); EXPECT_EQ("value", name1); #ifdef SQLITE_ENABLE_COLUMN_METADATA // Show how to get origin names of the table columns from which theses result columns come from. // Requires the SQLITE_ENABLE_COLUMN_METADATA preprocessor macro to be // also defined at compile times of the SQLite library itself. const std::string oname0 = query.getColumnOriginName(0); const std::string oname1 = query.getColumnOriginName(1); EXPECT_EQ("id", oname0); EXPECT_EQ("msg", oname1); #endif }