// Copyright 2016 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 <memory> #include <string> #include "fpdfsdk/cpdfsdk_helpers.h" #include "public/cpp/fpdf_scopers.h" #include "public/fpdf_edit.h" #include "public/fpdf_ppo.h" #include "public/fpdf_save.h" #include "public/fpdfview.h" #include "testing/embedder_test.h" #include "testing/gtest/include/gtest/gtest.h" #include "testing/test_support.h" namespace { class FPDFPPOEmbeddertest : public EmbedderTest {}; int FakeBlockWriter(FPDF_FILEWRITE* pThis, const void* pData, unsigned long size) { return size; } } // namespace TEST_F(FPDFPPOEmbeddertest, NoViewerPreferences) { EXPECT_TRUE(OpenDocument("hello_world.pdf")); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_FALSE(FPDF_CopyViewerPreferences(output_doc, document())); FPDF_CloseDocument(output_doc); } TEST_F(FPDFPPOEmbeddertest, ViewerPreferences) { EXPECT_TRUE(OpenDocument("viewer_ref.pdf")); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_TRUE(FPDF_CopyViewerPreferences(output_doc, document())); FPDF_CloseDocument(output_doc); } TEST_F(FPDFPPOEmbeddertest, ImportPages) { ASSERT_TRUE(OpenDocument("viewer_ref.pdf")); FPDF_PAGE page = LoadPage(0); EXPECT_TRUE(page); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); ASSERT_TRUE(output_doc); EXPECT_TRUE(FPDF_CopyViewerPreferences(output_doc, document())); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "1", 0)); EXPECT_EQ(1, FPDF_GetPageCount(output_doc)); FPDF_CloseDocument(output_doc); UnloadPage(page); } TEST_F(FPDFPPOEmbeddertest, ImportNPages) { ASSERT_TRUE(OpenDocument("rectangles_multi_pages.pdf")); ScopedFPDFDocument output_doc_2up( FPDF_ImportNPagesToOne(document(), 612, 792, 2, 1)); ASSERT_TRUE(output_doc_2up); EXPECT_EQ(3, FPDF_GetPageCount(output_doc_2up.get())); ScopedFPDFDocument output_doc_5up( FPDF_ImportNPagesToOne(document(), 612, 792, 5, 1)); ASSERT_TRUE(output_doc_5up); EXPECT_EQ(1, FPDF_GetPageCount(output_doc_5up.get())); ScopedFPDFDocument output_doc_8up( FPDF_ImportNPagesToOne(document(), 792, 612, 8, 1)); ASSERT_TRUE(output_doc_8up); EXPECT_EQ(1, FPDF_GetPageCount(output_doc_8up.get())); ScopedFPDFDocument output_doc_128up( FPDF_ImportNPagesToOne(document(), 792, 612, 128, 1)); ASSERT_TRUE(output_doc_128up); EXPECT_EQ(1, FPDF_GetPageCount(output_doc_128up.get())); } TEST_F(FPDFPPOEmbeddertest, BadNupParams) { ASSERT_TRUE(OpenDocument("rectangles_multi_pages.pdf")); FPDF_DOCUMENT output_doc_zero_row = FPDF_ImportNPagesToOne(document(), 612, 792, 0, 3); ASSERT_FALSE(output_doc_zero_row); FPDF_DOCUMENT output_doc_zero_col = FPDF_ImportNPagesToOne(document(), 612, 792, 2, 0); ASSERT_FALSE(output_doc_zero_col); FPDF_DOCUMENT output_doc_zero_width = FPDF_ImportNPagesToOne(document(), 0, 792, 2, 1); ASSERT_FALSE(output_doc_zero_width); FPDF_DOCUMENT output_doc_zero_height = FPDF_ImportNPagesToOne(document(), 612, 0, 7, 1); ASSERT_FALSE(output_doc_zero_height); } // TODO(Xlou): Add more tests to check output doc content of // FPDF_ImportNPagesToOne() TEST_F(FPDFPPOEmbeddertest, NupRenderImage) { ASSERT_TRUE(OpenDocument("rectangles_multi_pages.pdf")); const int kPageCount = 2; constexpr const char* kExpectedMD5s[kPageCount] = { "4d225b961da0f1bced7c83273e64c9b6", "fb18142190d770cfbc329d2b071aee4d"}; ScopedFPDFDocument output_doc_3up( FPDF_ImportNPagesToOne(document(), 792, 612, 3, 1)); ASSERT_TRUE(output_doc_3up); ASSERT_EQ(kPageCount, FPDF_GetPageCount(output_doc_3up.get())); for (int i = 0; i < kPageCount; ++i) { ScopedFPDFPage page(FPDF_LoadPage(output_doc_3up.get(), i)); ASSERT_TRUE(page); ScopedFPDFBitmap bitmap(RenderPageWithFlags(page.get(), nullptr, 0)); EXPECT_EQ(792, FPDFBitmap_GetWidth(bitmap.get())); EXPECT_EQ(612, FPDFBitmap_GetHeight(bitmap.get())); EXPECT_EQ(kExpectedMD5s[i], HashBitmap(bitmap.get())); } } TEST_F(FPDFPPOEmbeddertest, BadRepeatViewerPref) { ASSERT_TRUE(OpenDocument("repeat_viewer_ref.pdf")); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_TRUE(FPDF_CopyViewerPreferences(output_doc, document())); FPDF_FILEWRITE writer; writer.version = 1; writer.WriteBlock = FakeBlockWriter; EXPECT_TRUE(FPDF_SaveAsCopy(output_doc, &writer, 0)); FPDF_CloseDocument(output_doc); } TEST_F(FPDFPPOEmbeddertest, BadCircularViewerPref) { ASSERT_TRUE(OpenDocument("circular_viewer_ref.pdf")); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_TRUE(FPDF_CopyViewerPreferences(output_doc, document())); FPDF_FILEWRITE writer; writer.version = 1; writer.WriteBlock = FakeBlockWriter; EXPECT_TRUE(FPDF_SaveAsCopy(output_doc, &writer, 0)); FPDF_CloseDocument(output_doc); } TEST_F(FPDFPPOEmbeddertest, BadRanges) { EXPECT_TRUE(OpenDocument("hello_world.pdf")); FPDF_PAGE page = LoadPage(0); EXPECT_TRUE(page); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "clams", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "0", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "42", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "1,2", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "1-2", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), ",1", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "1,", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "1-", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "-1", 0)); EXPECT_FALSE(FPDF_ImportPages(output_doc, document(), "-,0,,,1-", 0)); FPDF_CloseDocument(output_doc); UnloadPage(page); } TEST_F(FPDFPPOEmbeddertest, GoodRanges) { EXPECT_TRUE(OpenDocument("viewer_ref.pdf")); FPDF_PAGE page = LoadPage(0); EXPECT_TRUE(page); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_TRUE(FPDF_CopyViewerPreferences(output_doc, document())); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "1,1,1,1", 0)); EXPECT_EQ(4, FPDF_GetPageCount(output_doc)); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "1-1", 0)); EXPECT_EQ(5, FPDF_GetPageCount(output_doc)); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "5-5", 0)); EXPECT_EQ(6, FPDF_GetPageCount(output_doc)); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "2-4", 0)); EXPECT_EQ(9, FPDF_GetPageCount(output_doc)); FPDF_CloseDocument(output_doc); UnloadPage(page); } TEST_F(FPDFPPOEmbeddertest, BUG_664284) { EXPECT_TRUE(OpenDocument("bug_664284.pdf")); FPDF_PAGE page = LoadPage(0); ASSERT_NE(nullptr, page); FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(output_doc); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "1", 0)); FPDF_CloseDocument(output_doc); UnloadPage(page); } TEST_F(FPDFPPOEmbeddertest, BUG_750568) { const char* const kHashes[] = { "64ad08132a1c5a166768298c8a578f57", "83b83e2f6bc80707d0a917c7634140b9", "913cd3723a451e4e46fbc2c05702d1ee", "81fb7cfd4860f855eb468f73dfeb6d60"}; ASSERT_TRUE(OpenDocument("bug_750568.pdf")); ASSERT_EQ(4, FPDF_GetPageCount(document())); for (size_t i = 0; i < 4; ++i) { FPDF_PAGE page = LoadPage(i); ASSERT_TRUE(page); ScopedFPDFBitmap bitmap = RenderLoadedPage(page); ASSERT_EQ(200, FPDFBitmap_GetWidth(bitmap.get())); ASSERT_EQ(200, FPDFBitmap_GetHeight(bitmap.get())); ASSERT_EQ(800, FPDFBitmap_GetStride(bitmap.get())); EXPECT_EQ(kHashes[i], HashBitmap(bitmap.get())); UnloadPage(page); } FPDF_DOCUMENT output_doc = FPDF_CreateNewDocument(); ASSERT_TRUE(output_doc); EXPECT_TRUE(FPDF_ImportPages(output_doc, document(), "1,2,3,4", 0)); ASSERT_EQ(4, FPDF_GetPageCount(output_doc)); for (size_t i = 0; i < 4; ++i) { FPDF_PAGE page = FPDF_LoadPage(output_doc, i); ASSERT_TRUE(page); ScopedFPDFBitmap bitmap = RenderPageWithFlags(page, nullptr, 0); ASSERT_EQ(200, FPDFBitmap_GetWidth(bitmap.get())); ASSERT_EQ(200, FPDFBitmap_GetHeight(bitmap.get())); ASSERT_EQ(800, FPDFBitmap_GetStride(bitmap.get())); EXPECT_EQ(kHashes[i], HashBitmap(bitmap.get())); FPDF_ClosePage(page); } FPDF_CloseDocument(output_doc); } TEST_F(FPDFPPOEmbeddertest, ImportWithZeroLengthStream) { EXPECT_TRUE(OpenDocument("zero_length_stream.pdf")); FPDF_PAGE page = LoadPage(0); ASSERT_TRUE(page); ScopedFPDFBitmap bitmap = RenderLoadedPage(page); ASSERT_EQ(200, FPDFBitmap_GetWidth(bitmap.get())); ASSERT_EQ(200, FPDFBitmap_GetHeight(bitmap.get())); ASSERT_EQ(800, FPDFBitmap_GetStride(bitmap.get())); std::string digest = HashBitmap(bitmap.get()); UnloadPage(page); FPDF_DOCUMENT new_doc = FPDF_CreateNewDocument(); EXPECT_TRUE(new_doc); EXPECT_TRUE(FPDF_ImportPages(new_doc, document(), "1", 0)); EXPECT_EQ(1, FPDF_GetPageCount(new_doc)); FPDF_PAGE new_page = FPDF_LoadPage(new_doc, 0); ASSERT_NE(nullptr, new_page); ScopedFPDFBitmap new_bitmap = RenderPageWithFlags(new_page, nullptr, 0); ASSERT_EQ(200, FPDFBitmap_GetWidth(new_bitmap.get())); ASSERT_EQ(200, FPDFBitmap_GetHeight(new_bitmap.get())); ASSERT_EQ(800, FPDFBitmap_GetStride(new_bitmap.get())); EXPECT_EQ(digest, HashBitmap(new_bitmap.get())); FPDF_ClosePage(new_page); FPDF_CloseDocument(new_doc); }