blob: f6daad6751e843fe3c37850b17f421fabc765ec0 [file] [log] [blame]
// Copyright 2016 The PDFium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <limits>
#include "core/fpdfapi/parser/cpdf_object.h"
#include "core/fpdfapi/parser/cpdf_parser.h"
#include "core/fpdfapi/parser/cpdf_syntax_parser.h"
#include "core/fxcrt/cfx_read_only_span_stream.h"
#include "core/fxcrt/fx_extension.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/utils/path_service.h"
using testing::ElementsAre;
using testing::IsEmpty;
TEST(SyntaxParserTest, ReadHexString) {
{
// Empty string.
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
pdfium::span<const uint8_t>()));
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
EXPECT_EQ(0, parser.GetPos());
}
{
// Blank string.
static const char data[] = " ";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
EXPECT_EQ(2, parser.GetPos());
}
{
// Skips unknown characters.
static const char data[] = "z12b";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x12, 0xb0));
EXPECT_EQ(4, parser.GetPos());
}
{
// Skips unknown characters.
static const char data[] = "*<&*#$^&@1";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x10));
EXPECT_EQ(10, parser.GetPos());
}
{
// Skips unknown characters.
static const char data[] = "\x80zab";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0xab));
EXPECT_EQ(4, parser.GetPos());
}
{
// Skips unknown characters.
static const char data[] = "\xffzab";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0xab));
EXPECT_EQ(4, parser.GetPos());
}
{
// Regular conversion.
static const char data[] = "1A2b>abcd";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x1a, 0x2b));
EXPECT_EQ(5, parser.GetPos());
}
{
// Position out of bounds.
static const char data[] = "12ab>";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
parser.SetPos(5);
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
parser.SetPos(6);
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
parser.SetPos(std::numeric_limits<FX_FILESIZE>::max());
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
// Check string still parses when set to 0.
parser.SetPos(0);
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x12, 0xab));
}
{
// Missing ending >.
static const char data[] = "1A2b";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x1a, 0x2b));
EXPECT_EQ(4, parser.GetPos());
}
{
// Missing ending >.
static const char data[] = "12abz";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x12, 0xab));
EXPECT_EQ(5, parser.GetPos());
}
{
// Uneven number of bytes.
static const char data[] = "1A2>asdf";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x1a, 0x20));
EXPECT_EQ(4, parser.GetPos());
}
{
// Uneven number of bytes.
static const char data[] = "1A2zasdf";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
EXPECT_THAT(parser.ReadHexString(), ElementsAre(0x1a, 0x2a, 0xdf));
EXPECT_EQ(8, parser.GetPos());
}
{
// Just ending character.
const char gt = '>';
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
pdfium::byte_span_from_ref(gt)));
EXPECT_THAT(parser.ReadHexString(), IsEmpty());
EXPECT_EQ(1, parser.GetPos());
}
}
TEST(SyntaxParserTest, GetInvalidReference) {
// Data with a reference with number CPDF_Object::kInvalidObjNum
static const char data[] = "4294967295 0 R";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(
ByteStringView(data).unsigned_span()));
RetainPtr<CPDF_Object> ref = parser.GetObjectBody(nullptr);
EXPECT_FALSE(ref);
}
TEST(SyntaxParserTest, PeekNextWord) {
static const uint8_t data[] = " WORD ";
CPDF_SyntaxParser parser(pdfium::MakeRetain<CFX_ReadOnlySpanStream>(data));
EXPECT_EQ("WORD", parser.PeekNextWord());
EXPECT_EQ("WORD", parser.GetNextWord().word);
}