blob: 45e2a5d4e60577b71ee386aa2f3d7e93f3108c19 [file] [log] [blame]
// Copyright 2015 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.
#ifndef TESTING_EMBEDDER_TEST_H_
#define TESTING_EMBEDDER_TEST_H_
#include <fstream>
#include <map>
#include <memory>
#include <string>
#include <vector>
#include "build/build_config.h"
#include "public/cpp/fpdf_scopers.h"
#include "public/fpdf_dataavail.h"
#include "public/fpdf_ext.h"
#include "public/fpdf_formfill.h"
#include "public/fpdf_save.h"
#include "public/fpdfview.h"
#include "testing/fake_file_access.h"
#include "testing/free_deleter.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/base/span.h"
#ifdef PDF_ENABLE_V8
namespace v8 {
class Platform;
#ifdef V8_USE_EXTERNAL_STARTUP_DATA
class StartupData;
#endif // V8_USE_EXTERNAL_STARTUP_DATA
} // namespace v8
#endif // PDF_ENABLE_V8
class TestLoader;
// The loading time of the CFGAS_FontMgr is linear in the number of times it is
// loaded. So, if a test suite has a lot of tests that need a font manager they
// can end up executing very, very slowly.
class EmbedderTestEnvironment final : public testing::Environment {
public:
explicit EmbedderTestEnvironment(const char* exe_path);
~EmbedderTestEnvironment() override;
// Note: does not create one if it does not exist.
static EmbedderTestEnvironment* GetInstance();
void SetUp() override;
void TearDown() override;
#ifdef PDF_ENABLE_V8
v8::Platform* platform() const { return platform_.get(); }
#endif // PDF_ENABLE_V8
private:
#ifdef PDF_ENABLE_V8
const char* const exe_path_;
#ifdef V8_USE_EXTERNAL_STARTUP_DATA
std::unique_ptr<v8::StartupData> v8_snapshot_;
#endif // V8_USE_EXTERNAL_STARTUP_DATA
std::unique_ptr<v8::Platform> platform_;
#endif // PDF_ENABLE_V8
};
// This class is used to load a PDF document, and then run programatic
// API tests against it.
class EmbedderTest : public ::testing::Test,
public UNSUPPORT_INFO,
public IPDF_JSPLATFORM,
public FPDF_FORMFILLINFO,
public FPDF_FILEWRITE {
public:
enum class LinearizeOption { kDefaultLinearize, kMustLinearize };
enum class JavaScriptOption { kDisableJavaScript, kEnableJavaScript };
class Delegate {
public:
virtual ~Delegate() = default;
// Equivalent to UNSUPPORT_INFO::FSDK_UnSupport_Handler().
virtual void UnsupportedHandler(int type) {}
// Equivalent to IPDF_JSPLATFORM::app_alert().
virtual int Alert(FPDF_WIDESTRING message,
FPDF_WIDESTRING title,
int type,
int icon) {
return 0;
}
// Equivalent to FPDF_FORMFILLINFO::FFI_SetTimer().
virtual int SetTimer(int msecs, TimerCallback fn) { return 0; }
// Equivalent to FPDF_FORMFILLINFO::FFI_KillTimer().
virtual void KillTimer(int id) {}
// Equivalent to FPDF_FORMFILLINFO::FFI_GetPage().
virtual FPDF_PAGE GetPage(FPDF_FORMFILLINFO* info,
FPDF_DOCUMENT document,
int page_index);
// Equivalent to FPDF_FORMFILLINFO::FFI_DoURIAction().
virtual void DoURIAction(FPDF_BYTESTRING uri) {}
// Equivalent to FPDF_FORMFILLINFO::FFI_DoGoToAction().
virtual void DoGoToAction(FPDF_FORMFILLINFO* info,
int page_index,
int zoom_mode,
float* pos_arry,
int array_size) {}
// Equivalent to FPDF_FORMFILLINFO::FFI_OnFocusChange().
virtual void OnFocusChange(FPDF_FORMFILLINFO* info,
FPDF_ANNOTATION annot,
int page_index) {}
// Equivalent to FPDF_FORMFILLINFO::FFI_DoURIActionWithKeyboardModifier().
virtual void DoURIActionWithKeyboardModifier(FPDF_FORMFILLINFO* info,
FPDF_BYTESTRING uri,
int modifiers) {}
};
EmbedderTest();
virtual ~EmbedderTest();
void SetUp() override;
void TearDown() override;
#ifdef PDF_ENABLE_V8
// Call before SetUp to pass shared isolate, otherwise PDFium creates one.
void SetExternalIsolate(void* isolate);
#endif // PDF_ENABLE_V8
void SetDelegate(Delegate* delegate) {
delegate_ = delegate ? delegate : default_delegate_.get();
}
void SetFormFillInfoVersion(int form_fill_info_version) {
form_fill_info_version_ = form_fill_info_version;
}
FPDF_DOCUMENT document() const { return document_; }
FPDF_FORMHANDLE form_handle() const { return form_handle_; }
// Create an empty document, and its form fill environment. Returns true
// on success or false on failure.
bool CreateEmptyDocument();
// Open the document specified by |filename|, and create its form fill
// environment, or return false on failure. The |filename| is relative to
// the test data directory where we store all the test files. |password| can
// be nullptr if the file is not password protected. If |javascript_opts|
// is kDisableJavascript, then the document will be given stubs in place
// of the real JS engine.
virtual bool OpenDocumentWithOptions(const std::string& filename,
const char* password,
LinearizeOption linearize_option,
JavaScriptOption javascript_option);
// Variants provided for convenience.
bool OpenDocument(const std::string& filename);
bool OpenDocumentLinearized(const std::string& filename);
bool OpenDocumentWithPassword(const std::string& filename,
const char* password);
bool OpenDocumentWithoutJavaScript(const std::string& filename);
// Close the document from a previous OpenDocument() call. This happens
// automatically at tear-down, and is usually not explicitly required,
// unless testing multiple documents or duplicate destruction.
void CloseDocument();
// Perform JavaScript actions that are to run at document open time.
void DoOpenActions();
// Determine the page numbers present in the document.
int GetFirstPageNum();
int GetPageCount();
// Load a specific page of the open document with a given non-negative
// |page_number|. On success, fire form events for the page and return a page
// handle. On failure, return nullptr.
// The caller does not own the returned page handle, but must call
// UnloadPage() on it when done.
// The caller cannot call this for a |page_number| if it already obtained and
// holds the page handle for that page.
FPDF_PAGE LoadPage(int page_number);
// Same as LoadPage(), but does not fire form events.
FPDF_PAGE LoadPageNoEvents(int page_number);
// Fire form unload events and release the resources for a |page| obtained
// from LoadPage(). Further use of |page| is prohibited after calling this.
void UnloadPage(FPDF_PAGE page);
// Same as UnloadPage(), but does not fire form events.
void UnloadPageNoEvents(FPDF_PAGE page);
// Apply standard highlighting color/alpha to forms.
void SetInitialFormFieldHighlight(FPDF_FORMHANDLE form);
// RenderLoadedPageWithFlags() with no flags.
ScopedFPDFBitmap RenderLoadedPage(FPDF_PAGE page);
// Convert |page| loaded via LoadPage() into a bitmap with the specified page
// rendering |flags|.
//
// See public/fpdfview.h for a list of page rendering flags.
ScopedFPDFBitmap RenderLoadedPageWithFlags(FPDF_PAGE page, int flags);
// RenderSavedPageWithFlags() with no flags.
ScopedFPDFBitmap RenderSavedPage(FPDF_PAGE page);
// Convert |page| loaded via LoadSavedPage() into a bitmap with the specified
// page rendering |flags|.
//
// See public/fpdfview.h for a list of page rendering flags.
ScopedFPDFBitmap RenderSavedPageWithFlags(FPDF_PAGE page, int flags);
// Convert |page| into a bitmap with the specified page rendering |flags|.
// The form handle associated with |page| should be passed in via |handle|.
// If |handle| is nullptr, then forms on the page will not be rendered.
//
// See public/fpdfview.h for a list of page rendering flags.
// If none of the above Render methods are appropriate, then use this one.
static ScopedFPDFBitmap RenderPageWithFlags(FPDF_PAGE page,
FPDF_FORMHANDLE handle,
int flags);
// Simplified form of RenderPageWithFlags() with no handle and no flags.
static ScopedFPDFBitmap RenderPage(FPDF_PAGE page);
#if defined(OS_WIN)
// Convert |page| into EMF with the specified page rendering |flags|.
static std::vector<uint8_t> RenderPageWithFlagsToEmf(FPDF_PAGE page,
int flags);
// Get the PostScript data from |emf_data|.
static std::string GetPostScriptFromEmf(pdfium::span<const uint8_t> emf_data);
#endif // defined(OS_WIN)
// Return bytes for each of the FPDFBitmap_* format types.
static int BytesPerPixelForFormat(int format);
protected:
using PageNumberToHandleMap = std::map<int, FPDF_PAGE>;
bool OpenDocumentHelper(const char* password,
LinearizeOption linearize_option,
JavaScriptOption javascript_option,
FakeFileAccess* network_simulator,
FPDF_DOCUMENT* document,
FPDF_AVAIL* avail,
FPDF_FORMHANDLE* form_handle);
FPDF_FORMHANDLE SetupFormFillEnvironment(FPDF_DOCUMENT doc,
JavaScriptOption javascript_option);
// Return the hash of only the pixels in |bitmap|. i.e. Excluding the gap, if
// any, at the end of a row where the stride is larger than width * bpp.
static std::string HashBitmap(FPDF_BITMAP bitmap);
#ifndef NDEBUG
// For debugging purposes.
// Write |bitmap| as a PNG to |filename|.
static void WriteBitmapToPng(FPDF_BITMAP bitmap, const std::string& filename);
#endif
// Check |bitmap| to make sure it has the right dimensions and content.
static void CompareBitmap(FPDF_BITMAP bitmap,
int expected_width,
int expected_height,
const char* expected_md5sum);
void ClearString() { data_string_.clear(); }
const std::string& GetString() const { return data_string_; }
static int GetBlockFromString(void* param,
unsigned long pos,
unsigned char* buf,
unsigned long size);
// See comments in the respective non-Saved versions of these methods.
FPDF_DOCUMENT OpenSavedDocument();
FPDF_DOCUMENT OpenSavedDocumentWithPassword(const char* password);
void CloseSavedDocument();
FPDF_PAGE LoadSavedPage(int page_number);
void CloseSavedPage(FPDF_PAGE page);
void VerifySavedRendering(FPDF_PAGE page,
int width,
int height,
const char* md5);
void VerifySavedDocument(int width, int height, const char* md5);
void SetWholeFileAvailable();
#ifndef NDEBUG
// For debugging purposes.
// While open, write any data that gets passed to WriteBlockCallback() to
// |filename|. This is typically used to capture data from FPDF_SaveAsCopy()
// calls.
void OpenPDFFileForWrite(const std::string& filename);
void ClosePDFFileForWrite();
#endif
std::unique_ptr<Delegate> default_delegate_;
Delegate* delegate_;
#ifdef PDF_ENABLE_XFA
int form_fill_info_version_ = 2;
#else // PDF_ENABLE_XFA
int form_fill_info_version_ = 1;
#endif // PDF_ENABLE_XFA
FPDF_DOCUMENT document_ = nullptr;
FPDF_FORMHANDLE form_handle_ = nullptr;
FPDF_AVAIL avail_ = nullptr;
FPDF_FILEACCESS file_access_; // must outlive |avail_|.
std::unique_ptr<FakeFileAccess> fake_file_access_; // must outlive |avail_|.
void* external_isolate_ = nullptr;
std::unique_ptr<TestLoader> loader_;
size_t file_length_ = 0;
std::unique_ptr<char, pdfium::FreeDeleter> file_contents_;
PageNumberToHandleMap page_map_;
FPDF_DOCUMENT saved_document_ = nullptr;
FPDF_FORMHANDLE saved_form_handle_ = nullptr;
FPDF_AVAIL saved_avail_ = nullptr;
FPDF_FILEACCESS saved_file_access_; // must outlive |saved_avail_|.
// must outlive |saved_avail_|.
std::unique_ptr<FakeFileAccess> saved_fake_file_access_;
PageNumberToHandleMap saved_page_map_;
private:
static void UnsupportedHandlerTrampoline(UNSUPPORT_INFO*, int type);
static int AlertTrampoline(IPDF_JSPLATFORM* plaform,
FPDF_WIDESTRING message,
FPDF_WIDESTRING title,
int type,
int icon);
static int SetTimerTrampoline(FPDF_FORMFILLINFO* info,
int msecs,
TimerCallback fn);
static void KillTimerTrampoline(FPDF_FORMFILLINFO* info, int id);
static FPDF_PAGE GetPageTrampoline(FPDF_FORMFILLINFO* info,
FPDF_DOCUMENT document,
int page_index);
static void DoURIActionTrampoline(FPDF_FORMFILLINFO* info,
FPDF_BYTESTRING uri);
static void DoGoToActionTrampoline(FPDF_FORMFILLINFO* info,
int page_index,
int zoom_mode,
float* pos_array,
int array_size);
static void OnFocusChangeTrampoline(FPDF_FORMFILLINFO* info,
FPDF_ANNOTATION annot,
int page_index);
static void DoURIActionWithKeyboardModifierTrampoline(FPDF_FORMFILLINFO* info,
FPDF_BYTESTRING uri,
int modifiers);
static int WriteBlockCallback(FPDF_FILEWRITE* pFileWrite,
const void* data,
unsigned long size);
// Helper method for the methods below.
static int GetPageNumberForPage(const PageNumberToHandleMap& page_map,
FPDF_PAGE page);
// Find |page| inside |page_map_| and return the associated page number, or -1
// if |page| cannot be found.
int GetPageNumberForLoadedPage(FPDF_PAGE page) const;
// Same as GetPageNumberForLoadedPage(), but with |saved_page_map_|.
int GetPageNumberForSavedPage(FPDF_PAGE page) const;
void UnloadPageCommon(FPDF_PAGE page, bool do_events);
FPDF_PAGE LoadPageCommon(int page_number, bool do_events);
std::string data_string_;
std::string saved_document_file_data_;
std::ofstream filestream_;
};
#endif // TESTING_EMBEDDER_TEST_H_