|  | // Copyright 2012 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "sql/statement.h" | 
|  |  | 
|  | #include <cstdint> | 
|  | #include <limits> | 
|  | #include <string> | 
|  | #include <string_view> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/containers/contains.h" | 
|  | #include "base/files/scoped_temp_dir.h" | 
|  | #include "base/strings/strcat.h" | 
|  | #include "base/strings/string_number_conversions.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "base/test/bind.h" | 
|  | #include "base/test/metrics/histogram_tester.h" | 
|  | #include "sql/database.h" | 
|  | #include "sql/test/scoped_error_expecter.h" | 
|  | #include "sql/test/test_helpers.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "third_party/sqlite/sqlite3.h" | 
|  |  | 
|  | namespace sql { | 
|  | namespace { | 
|  |  | 
|  | class StatementTest : public testing::Test { | 
|  | public: | 
|  | void SetUp() override { | 
|  | ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | 
|  | ASSERT_TRUE( | 
|  | db_.Open(temp_dir_.GetPath().AppendASCII("statement_test.sqlite"))); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | base::ScopedTempDir temp_dir_; | 
|  | Database db_{test::kTestTag}; | 
|  | }; | 
|  |  | 
|  | TEST_F(StatementTest, Assign) { | 
|  | Statement create; | 
|  | EXPECT_FALSE(create.is_valid()); | 
|  |  | 
|  | create.Assign(db_.GetUniqueStatement( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  | EXPECT_TRUE(create.is_valid()); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, Run) { | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  | ASSERT_TRUE(db_.Execute("INSERT INTO rows(a, b) VALUES(3, 12)")); | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT b FROM rows WHERE a=?")); | 
|  | EXPECT_FALSE(select.Succeeded()); | 
|  |  | 
|  | // Stepping it won't work since we haven't bound the value. | 
|  | EXPECT_FALSE(select.Step()); | 
|  |  | 
|  | // Run should fail since this produces output, and we should use Step(). This | 
|  | // gets a bit wonky since sqlite says this is OK so succeeded is set. | 
|  | select.Reset(/*clear_bound_vars=*/true); | 
|  | select.BindInt64(0, 3); | 
|  | EXPECT_FALSE(select.Run()); | 
|  | EXPECT_EQ(SQLITE_ROW, db_.GetErrorCode()); | 
|  | EXPECT_TRUE(select.Succeeded()); | 
|  |  | 
|  | // Resetting it should put it back to the previous state (not runnable). | 
|  | select.Reset(/*clear_bound_vars=*/true); | 
|  | EXPECT_FALSE(select.Succeeded()); | 
|  |  | 
|  | // Binding and stepping should produce one row. | 
|  | select.BindInt64(0, 3); | 
|  | EXPECT_TRUE(select.Step()); | 
|  | EXPECT_TRUE(select.Succeeded()); | 
|  | EXPECT_EQ(12, select.ColumnInt64(0)); | 
|  | EXPECT_FALSE(select.Step()); | 
|  | EXPECT_TRUE(select.Succeeded()); | 
|  | } | 
|  |  | 
|  | // Error callback called for error running a statement. | 
|  | TEST_F(StatementTest, DatabaseErrorCallbackCalledOnError) { | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  |  | 
|  | bool error_callback_called = false; | 
|  | int error = SQLITE_OK; | 
|  | db_.set_error_callback(base::BindLambdaForTesting( | 
|  | [&](int sqlite_error, sql::Statement* statement) { | 
|  | error_callback_called = true; | 
|  | error = sqlite_error; | 
|  | })); | 
|  |  | 
|  | // `rows` is a table with ROWID. https://www.sqlite.org/rowidtable.html | 
|  | // Since `a` is declared as INTEGER PRIMARY KEY, it is an alias for SQLITE's | 
|  | // rowid. This means `a` can only take on integer values. Attempting to insert | 
|  | // anything else causes the error callback handler to be called with | 
|  | // SQLITE_MISMATCH as error code. | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO rows(a) VALUES(?)")); | 
|  | ASSERT_TRUE(insert.is_valid()); | 
|  | insert.BindString(0, "not an integer, not suitable as primary key value"); | 
|  | EXPECT_FALSE(insert.Run()) | 
|  | << "Invalid statement should not Run() successfully"; | 
|  | EXPECT_TRUE(error_callback_called) | 
|  | << "Statement::Run() should report errors to the database error callback"; | 
|  | EXPECT_EQ(SQLITE_MISMATCH, error) | 
|  | << "Statement::Run() should report errors to the database error callback"; | 
|  | } | 
|  |  | 
|  | // Error expecter works for error running a statement. | 
|  | TEST_F(StatementTest, ScopedIgnoreError) { | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO rows(a) VALUES(?)")); | 
|  | EXPECT_TRUE(insert.is_valid()); | 
|  | insert.BindString(0, "not an integer, not suitable as primary key value"); | 
|  |  | 
|  | { | 
|  | sql::test::ScopedErrorExpecter expecter; | 
|  | expecter.ExpectError(SQLITE_MISMATCH); | 
|  | EXPECT_FALSE(insert.Run()); | 
|  | EXPECT_TRUE(expecter.SawExpectedErrors()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, Reset) { | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  | ASSERT_TRUE(db_.Execute("INSERT INTO rows(a, b) VALUES(3, 12)")); | 
|  | ASSERT_TRUE(db_.Execute("INSERT INTO rows(a, b) VALUES(4, 13)")); | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("SELECT b FROM rows WHERE a=?")); | 
|  | insert.BindInt64(0, 3); | 
|  | ASSERT_TRUE(insert.Step()); | 
|  | EXPECT_EQ(12, insert.ColumnInt64(0)); | 
|  | ASSERT_FALSE(insert.Step()); | 
|  |  | 
|  | insert.Reset(/*clear_bound_vars=*/false); | 
|  | // Verify that we can get all rows again. | 
|  | ASSERT_TRUE(insert.Step()); | 
|  | EXPECT_EQ(12, insert.ColumnInt64(0)); | 
|  | EXPECT_FALSE(insert.Step()); | 
|  |  | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | ASSERT_FALSE(insert.Step()); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BindInt64) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE ints(id INTEGER PRIMARY KEY, i INTEGER NOT NULL)")); | 
|  |  | 
|  | const std::vector<int64_t> values = { | 
|  | // Small positive values. | 
|  | 0, | 
|  | 1, | 
|  | 2, | 
|  | 10, | 
|  | 101, | 
|  | 1002, | 
|  |  | 
|  | // Small negative values. | 
|  | -1, | 
|  | -2, | 
|  | -3, | 
|  | -10, | 
|  | -101, | 
|  | -1002, | 
|  |  | 
|  | // Large values. | 
|  | std::numeric_limits<int64_t>::max(), | 
|  | std::numeric_limits<int64_t>::min(), | 
|  | }; | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO ints(i) VALUES(?)")); | 
|  | for (int64_t value : values) { | 
|  | insert.BindInt64(0, value); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | } | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT i FROM ints ORDER BY id")); | 
|  | for (int64_t value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | int64_t column_value = select.ColumnInt64(0); | 
|  | EXPECT_EQ(value, column_value); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Chrome features rely on being able to use uint64_t with ColumnInt64(). | 
|  | // This is supported, because (starting in C++20) casting between signed and | 
|  | // unsigned integers is well-defined in both directions. This test ensures that | 
|  | // the casting works as expected. | 
|  | TEST_F(StatementTest, BindInt64_FromUint64t) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | static constexpr char kSql[] = | 
|  | "CREATE TABLE ints(id INTEGER PRIMARY KEY NOT NULL, i INTEGER NOT NULL)"; | 
|  | ASSERT_TRUE(db_.Execute(kSql)); | 
|  |  | 
|  | const std::vector<uint64_t> values = { | 
|  | // Small positive values. | 
|  | 0, | 
|  | 1, | 
|  | 2, | 
|  | 10, | 
|  | 101, | 
|  | 1002, | 
|  |  | 
|  | // Large values. | 
|  | std::numeric_limits<int64_t>::max() - 1, | 
|  | std::numeric_limits<int64_t>::max(), | 
|  | std::numeric_limits<uint64_t>::max() - 1, | 
|  | std::numeric_limits<uint64_t>::max(), | 
|  | }; | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO ints(i) VALUES(?)")); | 
|  | for (uint64_t value : values) { | 
|  | insert.BindInt64(0, static_cast<int64_t>(value)); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | } | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT i FROM ints ORDER BY id")); | 
|  | for (uint64_t value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | int64_t column_value = select.ColumnInt64(0); | 
|  | uint64_t cast_column_value = static_cast<uint64_t>(column_value); | 
|  | EXPECT_EQ(value, cast_column_value) << " column_value: " << column_value; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BindBlob) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE blobs(id INTEGER PRIMARY KEY NOT NULL, b BLOB NOT NULL)")); | 
|  |  | 
|  | const std::vector<std::vector<uint8_t>> values = { | 
|  | {}, | 
|  | {0x01}, | 
|  | {0x41, 0x42, 0x43, 0x44}, | 
|  | }; | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO blobs(b) VALUES(?)")); | 
|  | for (const std::vector<uint8_t>& value : values) { | 
|  | insert.BindBlob(0, value); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | } | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT b FROM blobs ORDER BY id")); | 
|  | for (const std::vector<uint8_t>& value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | std::vector<uint8_t> column_value; | 
|  | EXPECT_TRUE(select.ColumnBlobAsVector(0, &column_value)); | 
|  | EXPECT_EQ(value, column_value); | 
|  | } | 
|  | EXPECT_FALSE(select.Step()); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BindBlob_String16Overload) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE blobs(id INTEGER PRIMARY KEY NOT NULL, b BLOB NOT NULL)")); | 
|  |  | 
|  | const std::vector<std::u16string> values = { | 
|  | std::u16string(), std::u16string(u"hello\n"), std::u16string(u"😀🍩🎉"), | 
|  | std::u16string(u"\xd800\xdc00text"),  // surrogate pair with text | 
|  | std::u16string(u"\xd8ff"),            // unpaired high surrogate | 
|  | std::u16string(u"\xdddd"),            // unpaired low surrogate | 
|  | std::u16string(u"\xdc00\xd800text"),  // lone low followed by lone high | 
|  | // surrogate and text | 
|  | std::u16string(1024, 0xdb23),         // long invalid UTF-16 | 
|  | }; | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO blobs(b) VALUES(?)")); | 
|  | for (const std::u16string& value : values) { | 
|  | insert.BindBlob(0, value); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | } | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT b FROM blobs ORDER BY id")); | 
|  | for (const std::u16string& value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | std::u16string column_value; | 
|  | EXPECT_TRUE(select.ColumnBlobAsString16(0, &column_value)); | 
|  | EXPECT_EQ(value, column_value); | 
|  | } | 
|  | EXPECT_FALSE(select.Step()); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BlobStressTest) { | 
|  | // Create a table that holds a whole lot of blobs. This could tickle | 
|  | // pointer-stability related bugs in the container that stores blob data | 
|  | // before it's being written. | 
|  | const int kMany = 200; | 
|  | std::string create_table_sql( | 
|  | "CREATE TABLE blobs(id INTEGER PRIMARY KEY NOT NULL "); | 
|  | for (int i = 0; i < kMany; ++i) { | 
|  | base::StrAppend(&create_table_sql, | 
|  | {", a", base::NumberToString(i), " BLOB NOT NULL"}); | 
|  | } | 
|  | create_table_sql.append(")"); | 
|  |  | 
|  | ASSERT_TRUE(db_.Execute(create_table_sql)); | 
|  |  | 
|  | std::vector<std::string> param_markers(kMany + 1, "?"); | 
|  | const std::string insert_sql = | 
|  | base::StrCat({"INSERT INTO blobs VALUES(", | 
|  | base::JoinString(param_markers, ", "), ")"}); | 
|  | sql::StatementID kInsertStatementId = SQL_FROM_HERE; | 
|  | { | 
|  | Statement insert(db_.GetCachedStatement(kInsertStatementId, insert_sql)); | 
|  | // ID row. | 
|  | insert.BindInt64(0, 1); | 
|  | for (int i = 0; i < kMany; ++i) { | 
|  | insert.BindBlob(i + 1, std::string(100, 'a' + i % 26)); | 
|  | } | 
|  |  | 
|  | // Make sure overwriting a blob works as expected. | 
|  | insert.BindBlob(50, std::string("overwrite")); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | } | 
|  |  | 
|  | // Verify the blobs read out as expected. | 
|  | { | 
|  | Statement select(db_.GetUniqueStatement("SELECT * FROM blobs")); | 
|  | ASSERT_TRUE(select.Step()); | 
|  | std::string output50, output51; | 
|  | EXPECT_TRUE(select.ColumnBlobAsString(50, &output50)); | 
|  | EXPECT_EQ("overwrite", output50); | 
|  | EXPECT_TRUE(select.ColumnBlobAsString(51, &output51)); | 
|  | EXPECT_EQ(std::string(100, 'y'), output51); | 
|  | } | 
|  |  | 
|  | // Make sure the underlying statement is reset i.e. the old bindings don't | 
|  | // persist across different invocations of `GetCachedStatement`. | 
|  | { | 
|  | Statement insert(db_.GetCachedStatement(kInsertStatementId, | 
|  | base::cstring_view(insert_sql))); | 
|  | // ID row. | 
|  | insert.BindInt64(0, 2); | 
|  | ASSERT_FALSE(insert.Run()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BindString) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE texts(id INTEGER PRIMARY KEY NOT NULL, t TEXT NOT NULL)")); | 
|  |  | 
|  | const std::vector<std::string> values = { | 
|  | "", | 
|  | "a", | 
|  | "\x01", | 
|  | std::string("\x00", 1), | 
|  | "abcd", | 
|  | "\x01\x02\x03\x04", | 
|  | std::string("\x01Test", 5), | 
|  | std::string("\x00Test", 5), | 
|  | }; | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO texts(t) VALUES(?)")); | 
|  | for (const std::string& value : values) { | 
|  | insert.BindString(0, value); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  | insert.Reset(/*clear_bound_vars=*/true); | 
|  | } | 
|  |  | 
|  | { | 
|  | Statement select(db_.GetUniqueStatement("SELECT t FROM texts ORDER BY id")); | 
|  | for (const std::string& value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | EXPECT_EQ(value, select.ColumnString(0)); | 
|  | } | 
|  | EXPECT_FALSE(select.Step()); | 
|  | } | 
|  |  | 
|  | { | 
|  | Statement select(db_.GetUniqueStatement("SELECT t FROM texts ORDER BY id")); | 
|  | for (const std::string& value : values) { | 
|  | ASSERT_TRUE(select.Step()); | 
|  | EXPECT_EQ(value, select.ColumnStringView(0)); | 
|  | } | 
|  | EXPECT_FALSE(select.Step()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, BindString_NullData) { | 
|  | // `id` makes SQLite's rowid mechanism explicit. We rely on it to retrieve | 
|  | // the rows in the same order that they were inserted. | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE texts(id INTEGER PRIMARY KEY NOT NULL, t TEXT NOT NULL)")); | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO texts(t) VALUES(?)")); | 
|  | insert.BindString(0, std::string_view(nullptr, 0)); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  |  | 
|  | Statement select(db_.GetUniqueStatement("SELECT t FROM texts ORDER BY id")); | 
|  | ASSERT_TRUE(select.Step()); | 
|  | EXPECT_EQ(std::string(), select.ColumnString(0)); | 
|  |  | 
|  | EXPECT_FALSE(select.Step()); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, GetSQLStatementExcludesBoundValues) { | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE texts(id INTEGER PRIMARY KEY NOT NULL, t TEXT NOT NULL)")); | 
|  |  | 
|  | Statement insert(db_.GetUniqueStatement("INSERT INTO texts(t) VALUES(?)")); | 
|  | insert.BindString(0, "John Doe"); | 
|  | ASSERT_TRUE(insert.Run()); | 
|  |  | 
|  | // Verify that GetSQLStatement doesn't leak any bound values that may be PII. | 
|  | std::string sql_statement = insert.GetSQLStatement(); | 
|  | EXPECT_TRUE(base::Contains(sql_statement, "INSERT INTO texts(t) VALUES(?)")); | 
|  | EXPECT_TRUE(base::Contains(sql_statement, "VALUES")); | 
|  | EXPECT_FALSE(base::Contains(sql_statement, "Doe")); | 
|  |  | 
|  | // Sanity check that the name was actually committed. | 
|  | Statement select(db_.GetUniqueStatement("SELECT t FROM texts ORDER BY id")); | 
|  | ASSERT_TRUE(select.Step()); | 
|  | EXPECT_EQ(select.ColumnString(0), "John Doe"); | 
|  | } | 
|  |  | 
|  | TEST_F(StatementTest, RunReportsPerformanceMetrics) { | 
|  | base::HistogramTester histogram_tester; | 
|  |  | 
|  | ASSERT_TRUE(db_.Execute( | 
|  | "CREATE TABLE rows(a INTEGER PRIMARY KEY NOT NULL, b INTEGER NOT NULL)")); | 
|  | ASSERT_TRUE(db_.Execute("INSERT INTO rows(a, b) VALUES(12, 42)")); | 
|  |  | 
|  | histogram_tester.ExpectTotalCount("Sql.Statement.Test.VMSteps", 0); | 
|  |  | 
|  | { | 
|  | Statement select(db_.GetUniqueStatement("SELECT b FROM rows WHERE a=?")); | 
|  | select.BindInt64(0, 12); | 
|  | ASSERT_TRUE(select.Step()); | 
|  | EXPECT_EQ(select.ColumnInt64(0), 42); | 
|  | } | 
|  |  | 
|  | histogram_tester.ExpectTotalCount("Sql.Statement.Test.VMSteps", 1); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace sql |