// Copyright 2017 PDFium 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 #include #include #include #include "core/fxcrt/fx_system.h" #include "public/cpp/fpdf_scopers.h" #include "public/fpdf_annot.h" #include "public/fpdf_edit.h" #include "public/fpdfview.h" #include "testing/embedder_test.h" #include "testing/fake_file_access.h" #include "testing/gmock/include/gmock/gmock-matchers.h" #include "testing/gtest/include/gtest/gtest.h" #include "testing/utils/path_service.h" namespace { class FileAccess final : public FPDF_FILEACCESS { public: explicit FileAccess(const std::string& file_name) { std::string file_path; if (!PathService::GetTestFilePath(file_name, &file_path)) return; file_contents_ = GetFileContents(file_path.c_str(), &file_length_); if (!file_contents_) return; m_FileLen = static_cast(file_length_); m_GetBlock = SGetBlock; m_Param = this; } private: int GetBlockImpl(unsigned long pos, unsigned char* pBuf, unsigned long size) { memcpy(pBuf, file_contents_.get() + pos, size); return size; } static int SGetBlock(void* param, unsigned long pos, unsigned char* pBuf, unsigned long size) { return static_cast(param)->GetBlockImpl(pos, pBuf, size); } size_t file_length_; std::unique_ptr file_contents_; }; } // namespace class CPDF_CreatorEmbedderTest : public EmbedderTest {}; TEST_F(CPDF_CreatorEmbedderTest, SavedDocsAreEqualAfterParse) { ASSERT_TRUE(OpenDocument("annotation_stamp_with_ap.pdf")); // Save without additional data reading. EXPECT_TRUE(FPDF_SaveAsCopy(document(), this, 0)); const std::string saved_doc_1 = GetString(); ClearString(); { // Do some read only operations. ASSERT_GE(1, FPDF_GetPageCount(document())); FPDF_PAGE page = LoadPage(0); ASSERT_TRUE(page); ScopedFPDFBitmap bitmap = RenderLoadedPageWithFlags(page, FPDF_ANNOT); EXPECT_EQ(595, FPDFBitmap_GetWidth(bitmap.get())); EXPECT_EQ(842, FPDFBitmap_GetHeight(bitmap.get())); UnloadPage(page); } // Save when we have additional loaded data. EXPECT_TRUE(FPDF_SaveAsCopy(document(), this, 0)); const std::string saved_doc_2 = GetString(); ClearString(); // The sizes of saved docs should be equal. EXPECT_EQ(saved_doc_1.size(), saved_doc_2.size()); } TEST_F(CPDF_CreatorEmbedderTest, BUG_873) { EXPECT_TRUE(OpenDocument("embedded_attachments.pdf")); EXPECT_TRUE(FPDF_SaveAsCopy(document(), this, 0)); // Cannot match second part of the ID since it is randomly generated. std::string saved_data = GetString(); const char kTrailerBeforeSecondID[] = "trailer\r\n<<"; ASSERT_THAT(saved_data, testing::HasSubstr(kTrailerBeforeSecondID)); size_t trailer_start = saved_data.find(kTrailerBeforeSecondID); constexpr size_t kIdLen = 32; size_t trailer_continuation = trailer_start + strlen(kTrailerBeforeSecondID) + kIdLen; std::string data_after_second_id = saved_data.substr(trailer_continuation); EXPECT_THAT(data_after_second_id, testing::StartsWith(">]>>\r\n")); } TEST_F(CPDF_CreatorEmbedderTest, SaveLinearizedInfo) { FileAccess file_acc("linearized.pdf"); FakeFileAccess fake_acc(&file_acc); avail_ = FPDFAvail_Create(fake_acc.GetFileAvail(), fake_acc.GetFileAccess()); while (PDF_DATA_AVAIL != FPDFAvail_IsDocAvail(avail_, fake_acc.GetDownloadHints())) { fake_acc.SetRequestedDataAvailable(); } document_ = FPDFAvail_GetDocument(avail_, nullptr); ASSERT_TRUE(document_); // Load second page, to parse additional crossref sections. while (PDF_DATA_AVAIL != FPDFAvail_IsPageAvail(avail_, 1, fake_acc.GetDownloadHints())) { fake_acc.SetRequestedDataAvailable(); } // Simulate downloading of whole file. fake_acc.SetWholeFileAvailable(); // Save document. EXPECT_TRUE(FPDF_SaveAsCopy(document_, this, 0)); const std::string saved_doc = GetString(); EXPECT_THAT(saved_doc, ::testing::HasSubstr("/Info")); }