| // Copyright 2016 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/ntp_snippets/remote/ntp_snippets_database.h" |
| |
| #include <memory> |
| |
| #include "base/bind.h" |
| #include "base/callback.h" |
| #include "base/files/file_path.h" |
| #include "base/files/scoped_temp_dir.h" |
| #include "base/macros.h" |
| #include "base/message_loop/message_loop.h" |
| #include "base/run_loop.h" |
| #include "base/threading/thread_task_runner_handle.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| using testing::ElementsAre; |
| using testing::IsEmpty; |
| using testing::Mock; |
| using testing::_; |
| |
| namespace ntp_snippets { |
| |
| bool operator==(const SnippetSource& lhs, const SnippetSource& rhs) { |
| return lhs.url == rhs.url && lhs.publisher_name == rhs.publisher_name && |
| lhs.amp_url == rhs.amp_url; |
| } |
| |
| bool operator==(const NTPSnippet& lhs, const NTPSnippet& rhs) { |
| return lhs.id() == rhs.id() && lhs.title() == rhs.title() && |
| lhs.snippet() == rhs.snippet() && |
| lhs.salient_image_url() == rhs.salient_image_url() && |
| lhs.publish_date() == rhs.publish_date() && |
| lhs.expiry_date() == rhs.expiry_date() && |
| lhs.source_index() == rhs.source_index() && |
| lhs.sources() == rhs.sources() && lhs.score() == rhs.score() && |
| lhs.is_dismissed() == rhs.is_dismissed(); |
| } |
| |
| namespace { |
| |
| std::unique_ptr<NTPSnippet> CreateTestSnippet() { |
| std::unique_ptr<NTPSnippet> snippet(new NTPSnippet("http://localhost")); |
| snippet->add_source( |
| SnippetSource(GURL("http://localhost"), "Publisher", GURL("http://amp"))); |
| return snippet; |
| } |
| |
| MATCHER_P(SnippetEq, snippet, "") { |
| return *arg == *snippet; |
| } |
| |
| } // namespace |
| |
| class NTPSnippetsDatabaseTest : public testing::Test { |
| public: |
| NTPSnippetsDatabaseTest() { |
| EXPECT_TRUE(database_dir_.CreateUniqueTempDir()); |
| } |
| |
| ~NTPSnippetsDatabaseTest() override { |
| // We need to run the message loop after deleting the database, because |
| // ProtoDatabaseImpl deletes the actual LevelDB asynchronously on the task |
| // runner. Without this, we'd get reports of memory leaks. |
| db_.reset(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| void CreateDatabase() { |
| // Explicitly destroy any existing database first, so it releases the lock |
| // on the file. |
| db_.reset(); |
| |
| db_.reset(new NTPSnippetsDatabase(database_dir_.GetPath(), |
| base::ThreadTaskRunnerHandle::Get())); |
| } |
| |
| NTPSnippetsDatabase* db() { return db_.get(); } |
| |
| void OnSnippetsLoaded(NTPSnippet::PtrVector snippets) { |
| OnSnippetsLoadedImpl(snippets); |
| } |
| MOCK_METHOD1(OnSnippetsLoadedImpl, |
| void(const NTPSnippet::PtrVector& snippets)); |
| |
| MOCK_METHOD1(OnImageLoaded, void(std::string)); |
| |
| private: |
| base::MessageLoop message_loop_; |
| base::ScopedTempDir database_dir_; |
| std::unique_ptr<NTPSnippetsDatabase> db_; |
| |
| DISALLOW_COPY_AND_ASSIGN(NTPSnippetsDatabaseTest); |
| }; |
| |
| TEST_F(NTPSnippetsDatabaseTest, Init) { |
| ASSERT_FALSE(db()); |
| |
| CreateDatabase(); |
| EXPECT_FALSE(db()->IsInitialized()); |
| |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_TRUE(db()->IsInitialized()); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, LoadBeforeInit) { |
| CreateDatabase(); |
| EXPECT_FALSE(db()->IsInitialized()); |
| |
| // Start a snippet and image load before the DB is initialized. |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| db()->LoadImage("id", base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| |
| // They should be serviced once initialization finishes. |
| EXPECT_CALL(*this, OnSnippetsLoadedImpl(_)); |
| EXPECT_CALL(*this, OnImageLoaded(_)); |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_TRUE(db()->IsInitialized()); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, LoadAfterInit) { |
| CreateDatabase(); |
| EXPECT_FALSE(db()->IsInitialized()); |
| |
| EXPECT_CALL(*this, OnSnippetsLoadedImpl(_)).Times(0); |
| base::RunLoop().RunUntilIdle(); |
| EXPECT_TRUE(db()->IsInitialized()); |
| |
| Mock::VerifyAndClearExpectations(this); |
| |
| EXPECT_CALL(*this, OnSnippetsLoadedImpl(_)); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| EXPECT_CALL(*this, OnImageLoaded(_)); |
| db()->LoadImage("id", base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, Save) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| std::string image_data("pretty image"); |
| |
| // Store a snippet and an image. |
| db()->SaveSnippet(*snippet); |
| db()->SaveImage(snippet->id(), image_data); |
| |
| // Make sure they're there. |
| EXPECT_CALL(*this, |
| OnSnippetsLoadedImpl(ElementsAre(SnippetEq(snippet.get())))); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| |
| Mock::VerifyAndClearExpectations(this); |
| |
| EXPECT_CALL(*this, OnImageLoaded(image_data)); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, SavePersist) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| std::string image_data("pretty image"); |
| |
| // Store a snippet and an image. |
| db()->SaveSnippet(*snippet); |
| db()->SaveImage(snippet->id(), image_data); |
| base::RunLoop().RunUntilIdle(); |
| |
| // They should still exist after recreating the database. |
| CreateDatabase(); |
| |
| EXPECT_CALL(*this, |
| OnSnippetsLoadedImpl(ElementsAre(SnippetEq(snippet.get())))); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| EXPECT_CALL(*this, OnImageLoaded(image_data)); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, Update) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| |
| // Store a snippet. |
| db()->SaveSnippet(*snippet); |
| |
| // Change it. |
| const std::string text("some text"); |
| snippet->set_snippet(text); |
| db()->SaveSnippet(*snippet); |
| |
| // Make sure we get the updated version. |
| EXPECT_CALL(*this, |
| OnSnippetsLoadedImpl(ElementsAre(SnippetEq(snippet.get())))); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, Delete) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| |
| // Store a snippet. |
| db()->SaveSnippet(*snippet); |
| |
| // Make sure it's there. |
| EXPECT_CALL(*this, |
| OnSnippetsLoadedImpl(ElementsAre(SnippetEq(snippet.get())))); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Delete the snippet. |
| db()->DeleteSnippet(snippet->id()); |
| |
| // Make sure it's gone. |
| EXPECT_CALL(*this, OnSnippetsLoadedImpl(IsEmpty())); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, DeleteSnippetDoesNotDeleteImage) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| std::string image_data("pretty image"); |
| |
| // Store a snippet and image. |
| db()->SaveSnippet(*snippet); |
| db()->SaveImage(snippet->id(), image_data); |
| base::RunLoop().RunUntilIdle(); |
| |
| // Make sure they're there. |
| EXPECT_CALL(*this, |
| OnSnippetsLoadedImpl(ElementsAre(SnippetEq(snippet.get())))); |
| db()->LoadSnippets(base::Bind(&NTPSnippetsDatabaseTest::OnSnippetsLoaded, |
| base::Unretained(this))); |
| EXPECT_CALL(*this, OnImageLoaded(image_data)); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Delete the snippet. |
| db()->DeleteSnippet(snippet->id()); |
| |
| // Make sure the image is still there. |
| EXPECT_CALL(*this, OnImageLoaded(image_data)); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(NTPSnippetsDatabaseTest, DeleteImage) { |
| CreateDatabase(); |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(db()->IsInitialized()); |
| |
| std::unique_ptr<NTPSnippet> snippet = CreateTestSnippet(); |
| std::string image_data("pretty image"); |
| |
| // Store the image. |
| db()->SaveImage(snippet->id(), image_data); |
| base::RunLoop().RunUntilIdle(); |
| |
| // Make sure the image is there. |
| EXPECT_CALL(*this, OnImageLoaded(image_data)); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| |
| Mock::VerifyAndClearExpectations(this); |
| |
| // Delete the snippet. |
| db()->DeleteImage(snippet->id()); |
| |
| // Make sure the image is gone. |
| EXPECT_CALL(*this, OnImageLoaded(std::string())); |
| db()->LoadImage(snippet->id(), |
| base::Bind(&NTPSnippetsDatabaseTest::OnImageLoaded, |
| base::Unretained(this))); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| } // namespace ntp_snippets |