1 Star 0 Fork 0

广州捷仓科技有限公司/gofpdi

加入 Gitee
与超过 1400万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
reader.go 44.14 KB
一键复制 编辑 原始数据 按行查看 历史
jiecang 提交于 2024-10-08 09:27 +08:00 . 开放GetPageRotation函数
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734
package gofpdi
import (
"bufio"
"bytes"
"compress/zlib"
"encoding/ascii85"
"encoding/binary"
"fmt"
"io"
"io/ioutil"
"math"
"os"
"strconv"
"strings"
"github.com/pkg/errors"
)
type PdfReader struct {
availableBoxes []string
stack []string
trailer *PdfValue
catalog *PdfValue
pages []*PdfValue
xrefPos int
xref map[int]map[int]int
xrefStream map[int][2]int
f io.ReadSeeker
nBytes int64
sourceFile string
curPage int
alreadyRead bool
pageCount int
}
type Ascii85DecodeError struct {
Message string
}
func (e *Ascii85DecodeError) Error() string {
return fmt.Sprintf("error decoding ascii85: %s", e.Message)
}
func NewPdfReaderFromStream(sourceFile string, rs io.ReadSeeker) (*PdfReader, error) {
length, err := rs.Seek(0, 2)
if err != nil {
return nil, errors.Wrapf(err, "Failed to determine stream length")
}
parser := &PdfReader{f: rs, sourceFile: sourceFile, nBytes: length}
if err := parser.init(); err != nil {
return nil, errors.Wrap(err, "Failed to initialize parser")
}
if err := parser.read(); err != nil {
return nil, errors.Wrap(err, "Failed to read pdf from stream")
}
return parser, nil
}
func NewPdfReader(filename string) (*PdfReader, error) {
var err error
f, err := os.Open(filename)
if err != nil {
return nil, errors.Wrap(err, "Failed to open file")
}
info, err := f.Stat()
if err != nil {
return nil, errors.Wrap(err, "Failed to obtain file information")
}
parser := &PdfReader{f: f, sourceFile: filename, nBytes: info.Size()}
if err = parser.init(); err != nil {
return nil, errors.Wrap(err, "Failed to initialize parser")
}
if err = parser.read(); err != nil {
return nil, errors.Wrap(err, "Failed to read pdf")
}
return parser, nil
}
func (this *PdfReader) init() error {
this.availableBoxes = []string{"/MediaBox", "/CropBox", "/BleedBox", "/TrimBox", "/ArtBox"}
this.xref = make(map[int]map[int]int, 0)
this.xrefStream = make(map[int][2]int, 0)
err := this.read()
if err != nil {
return errors.Wrap(err, "Failed to read pdf")
}
return nil
}
type PdfValue struct {
Type int
String string
Token string
Int int
Real float64
Bool bool
Dictionary map[string]*PdfValue
Array []*PdfValue
Id int
NewId int
Gen int
Value *PdfValue
Stream *PdfValue
Bytes []byte
}
// Jump over comments
func (this *PdfReader) skipComments(r *bufio.Reader) error {
var err error
var b byte
for {
b, err = r.ReadByte()
if err != nil {
if err == io.EOF {
break
}
return errors.Wrap(err, "Failed to ReadByte while skipping comments")
}
if b == '\n' || b == '\r' {
if b == '\r' {
// Peek and see if next char is \n
b2, err := r.ReadByte()
if err != nil {
return errors.Wrap(err, "Failed to read byte")
}
if b2 != '\n' {
r.UnreadByte()
}
}
break
}
}
return nil
}
// Advance reader so that whitespace is ignored
func (this *PdfReader) skipWhitespace(r *bufio.Reader) error {
var err error
var b byte
for {
b, err = r.ReadByte()
if err != nil {
if err == io.EOF {
break
}
return errors.Wrap(err, "Failed to read byte")
}
if b == ' ' || b == '\n' || b == '\r' || b == '\t' {
continue
} else {
r.UnreadByte()
break
}
}
return nil
}
// Read a token
func (this *PdfReader) readToken(r *bufio.Reader) (string, error) {
var err error
// If there is a token available on the stack, pop it out and return it.
if len(this.stack) > 0 {
var popped string
popped, this.stack = this.stack[len(this.stack)-1], this.stack[:len(this.stack)-1]
return popped, nil
}
err = this.skipWhitespace(r)
if err != nil {
return "", errors.Wrap(err, "Failed to skip whitespace")
}
b, err := r.ReadByte()
if err != nil {
if err == io.EOF {
return "", nil
}
return "", errors.Wrap(err, "Failed to read byte")
}
switch b {
case '[', ']', '(', ')':
// This is either an array or literal string delimeter, return it.
return string(b), nil
case '<', '>':
// This could either be a hex string or a dictionary delimiter.
// Determine the appropriate case and return the token.
nb, err := r.ReadByte()
if err != nil {
if err == io.EOF {
return string(b), nil
}
return "", errors.Wrap(err, "Failed to read byte")
}
if nb == b {
return string(b) + string(nb), nil
} else {
r.UnreadByte()
return string(b), nil
}
case '%':
err = this.skipComments(r)
if err != nil {
return "", errors.Wrap(err, "Failed to skip comments")
}
return this.readToken(r)
default:
// FIXME this may not be performant to create new strings for each byte
// Is it probably better to create a buffer and then convert to a string at the end.
str := string(b)
loop:
for {
b, err := r.ReadByte()
if err != nil {
if err == io.EOF {
break loop
}
return "", errors.Wrap(err, "Failed to read byte")
}
switch b {
case ' ', '%', '[', ']', '<', '>', '(', ')', '\r', '\n', '\t', '/':
r.UnreadByte()
break loop
default:
str += string(b)
}
}
return str, nil
}
}
// Read a value based on a token
func (this *PdfReader) readValue(r *bufio.Reader, t string) (*PdfValue, error) {
var err error
var b byte
result := &PdfValue{}
result.Type = -1
result.Token = t
result.Dictionary = make(map[string]*PdfValue, 0)
result.Array = make([]*PdfValue, 0)
switch t {
case "<":
// This is a hex string
// Read bytes until '>' is found
var s string
for {
b, err = r.ReadByte()
if err != nil {
return nil, errors.Wrap(err, "Failed to read byte")
}
if b != '>' {
s += string(b)
} else {
break
}
}
result.Type = PDF_TYPE_HEX
result.String = s
case "<<":
// This is a dictionary
// Recurse into this function until we reach the end of the dictionary.
for {
key, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
if key == "" {
return nil, errors.New("Token is empty")
}
if key == ">>" {
break
}
// read next token
newKey, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
value, err := this.readValue(r, newKey)
if err != nil {
return nil, errors.Wrap(err, "Failed to read value for token: "+newKey)
}
if value.Type == -1 {
return result, nil
}
// Catch missing value
if value.Type == PDF_TYPE_TOKEN && value.String == ">>" {
result.Type = PDF_TYPE_NULL
result.Dictionary[key] = value
break
}
// Set value in dictionary
result.Dictionary[key] = value
}
result.Type = PDF_TYPE_DICTIONARY
return result, nil
case "[":
// This is an array
tmpResult := make([]*PdfValue, 0)
// Recurse into this function until we reach the end of the array
for {
key, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
if key == "" {
return nil, errors.New("Token is empty")
}
if key == "]" {
break
}
value, err := this.readValue(r, key)
if err != nil {
return nil, errors.Wrap(err, "Failed to read value for token: "+key)
}
if value.Type == -1 {
return result, nil
}
tmpResult = append(tmpResult, value)
}
result.Type = PDF_TYPE_ARRAY
result.Array = tmpResult
case "(":
// This is a string
openBrackets := 1
// Create new buffer
var buf bytes.Buffer
// Read bytes until brackets are balanced
for openBrackets > 0 {
b, err := r.ReadByte()
if err != nil {
return nil, errors.Wrap(err, "Failed to read byte")
}
switch b {
case '(':
openBrackets++
case ')':
openBrackets--
case '\\':
nb, err := r.ReadByte()
if err != nil {
return nil, errors.Wrap(err, "Failed to read byte")
}
buf.WriteByte(b)
buf.WriteByte(nb)
continue
}
if openBrackets > 0 {
buf.WriteByte(b)
}
}
result.Type = PDF_TYPE_STRING
result.String = buf.String()
case "stream":
return nil, errors.New("Stream not implemented")
default:
result.Type = PDF_TYPE_TOKEN
result.Token = t
if is_numeric(t) {
// A numeric token. Make sure that it is not part of something else
t2, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
if t2 != "" {
if is_numeric(t2) {
// Two numeric tokens in a row.
// In this case, we're probably in front of either an object reference
// or an object specification.
// Determine the case and return the data.
t3, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
if t3 != "" {
switch t3 {
case "obj":
result.Type = PDF_TYPE_OBJDEC
result.Id, _ = strconv.Atoi(t)
result.Gen, _ = strconv.Atoi(t2)
return result, nil
case "R":
result.Type = PDF_TYPE_OBJREF
result.Id, _ = strconv.Atoi(t)
result.Gen, _ = strconv.Atoi(t2)
return result, nil
}
// If we get to this point, that numeric value up there was just a numeric value.
// Push the extra tokens back into the stack and return the value.
this.stack = append(this.stack, t3)
}
}
this.stack = append(this.stack, t2)
}
if n, err := strconv.Atoi(t); err == nil {
result.Type = PDF_TYPE_NUMERIC
result.Int = n
result.Real = float64(n) // Also assign Real value here to fix page box parsing bugs
} else {
result.Type = PDF_TYPE_REAL
result.Real, _ = strconv.ParseFloat(t, 64)
}
} else if t == "true" || t == "false" {
result.Type = PDF_TYPE_BOOLEAN
result.Bool = t == "true"
} else if t == "null" {
result.Type = PDF_TYPE_NULL
} else {
result.Type = PDF_TYPE_TOKEN
result.Token = t
}
}
return result, nil
}
// Resolve a compressed object (PDF 1.5)
func (this *PdfReader) resolveCompressedObject(objSpec *PdfValue) (*PdfValue, error) {
var err error
// Make sure object reference exists in xrefStream
if _, ok := this.xrefStream[objSpec.Id]; !ok {
return nil, errors.New(fmt.Sprintf("Could not find object ID %d in xref stream or xref table.", objSpec.Id))
}
// Get object id and index
objectId := this.xrefStream[objSpec.Id][0]
objectIndex := this.xrefStream[objSpec.Id][1]
// Read compressed object
compressedObjSpec := &PdfValue{Type: PDF_TYPE_OBJREF, Id: objectId, Gen: 0}
// Resolve compressed object
compressedObj, err := this.resolveObject(compressedObjSpec)
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve compressed object")
}
// Verify object type is /ObjStm
if _, ok := compressedObj.Value.Dictionary["/Type"]; ok {
if compressedObj.Value.Dictionary["/Type"].Token != "/ObjStm" {
return nil, errors.New("Expected compressed object type to be /ObjStm")
}
} else {
return nil, errors.New("Could not determine compressed object type.")
}
// Get number of sub-objects in compressed object
n := compressedObj.Value.Dictionary["/N"].Int
if n <= 0 {
return nil, errors.New("No sub objects in compressed object")
}
// Get offset of first object
first := compressedObj.Value.Dictionary["/First"].Int
// Get length
//length := compressedObj.Value.Dictionary["/Length"].Int
// Check for filter
filter := ""
if _, ok := compressedObj.Value.Dictionary["/Filter"]; ok {
filter = compressedObj.Value.Dictionary["/Filter"].Token
if filter != "/FlateDecode" {
return nil, errors.New("Unsupported filter - expected /FlateDecode, got: " + filter)
}
}
if filter == "/FlateDecode" {
// Decompress if filter is /FlateDecode
// Uncompress zlib compressed data
var out bytes.Buffer
zlibReader, _ := zlib.NewReader(bytes.NewBuffer(compressedObj.Stream.Bytes))
defer zlibReader.Close()
io.Copy(&out, zlibReader)
// Set stream to uncompressed data
compressedObj.Stream.Bytes = out.Bytes()
}
// Get io.Reader for bytes
r := bufio.NewReader(bytes.NewBuffer(compressedObj.Stream.Bytes))
subObjId := 0
subObjPos := 0
// Read sub-object indeces and their positions within the (un)compressed object
for i := 0; i < n; i++ {
var token string
var _objidx int
var _objpos int
// Read first token (object index)
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
// Convert line (string) into int
_objidx, err = strconv.Atoi(token)
if err != nil {
return nil, errors.Wrap(err, "Failed to convert token into integer: "+token)
}
// Read first token (object index)
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
// Convert line (string) into int
_objpos, err = strconv.Atoi(token)
if err != nil {
return nil, errors.Wrap(err, "Failed to convert token into integer: "+token)
}
if i == objectIndex {
subObjId = _objidx
subObjPos = _objpos
}
}
// Now create an io.ReadSeeker
rs := io.ReadSeeker(bytes.NewReader(compressedObj.Stream.Bytes))
// Determine where to seek to (sub-object position + /First)
seekTo := int64(subObjPos + first)
// Fast forward to the object
rs.Seek(seekTo, 0)
// Create a new io.Reader
r = bufio.NewReader(rs)
// Read token
token, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
// Read object
obj, err := this.readValue(r, token)
if err != nil {
return nil, errors.Wrap(err, "Failed to read value for token: "+token)
}
result := &PdfValue{}
result.Id = subObjId
result.Gen = 0
result.Type = PDF_TYPE_OBJECT
result.Value = obj
return result, nil
}
func (this *PdfReader) resolveObject(objSpec *PdfValue) (*PdfValue, error) {
var err error
var old_pos int64
// Create new bufio.Reader
r := bufio.NewReader(this.f)
if objSpec.Type == PDF_TYPE_OBJREF {
// This is a reference, resolve it.
offset := this.xref[objSpec.Id][objSpec.Gen]
if _, ok := this.xref[objSpec.Id]; !ok {
// This may be a compressed object
return this.resolveCompressedObject(objSpec)
}
// Save current file position
// This is needed if you want to resolve reference while you're reading another object.
// (e.g.: if you need to determine the length of a stream)
old_pos, err = this.f.Seek(0, os.SEEK_CUR)
if err != nil {
return nil, errors.Wrap(err, "Failed to get current position of file")
}
// Reposition the file pointer and load the object header
_, err = this.f.Seek(int64(offset), 0)
if err != nil {
return nil, errors.Wrap(err, "Failed to set position of file")
}
token, err := this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
obj, err := this.readValue(r, token)
if err != nil {
return nil, errors.Wrap(err, "Failed to read value for token: "+token)
}
if obj.Type != PDF_TYPE_OBJDEC {
return nil, errors.New(fmt.Sprintf("Expected type to be PDF_TYPE_OBJDEC, got: %d", obj.Type))
}
if obj.Id != objSpec.Id {
return nil, errors.New(fmt.Sprintf("Object ID (%d) does not match ObjSpec ID (%d)", obj.Id, objSpec.Id))
}
if obj.Gen != objSpec.Gen {
return nil, errors.New("Object Gen does not match ObjSpec Gen")
}
// Read next token
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
// Read actual object value
value, err := this.readValue(r, token)
if err != nil {
return nil, errors.Wrap(err, "Failed to read value for token: "+token)
}
// Read next token
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
result := &PdfValue{}
result.Id = obj.Id
result.Gen = obj.Gen
result.Type = PDF_TYPE_OBJECT
result.Value = value
if token == "stream" {
result.Type = PDF_TYPE_STREAM
err = this.skipWhitespace(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to skip whitespace")
}
// Get stream length dictionary
lengthDict := value.Dictionary["/Length"]
// Get number of bytes of stream
length := lengthDict.Int
// If lengthDict is an object reference, resolve the object and set length
if lengthDict.Type == PDF_TYPE_OBJREF {
lengthDict, err = this.resolveObject(lengthDict)
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve length object of stream")
}
// Set length to resolved object value
length = lengthDict.Value.Int
}
// Read length bytes
bytes := make([]byte, length)
// Cannot use reader.Read() because that may not read all the bytes
_, err := io.ReadFull(r, bytes)
if err != nil {
return nil, errors.Wrap(err, "Failed to read bytes from buffer")
}
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
if token != "endstream" {
return nil, errors.New("Expected next token to be: endstream, got: " + token)
}
token, err = this.readToken(r)
if err != nil {
return nil, errors.Wrap(err, "Failed to read token")
}
streamObj := &PdfValue{}
streamObj.Type = PDF_TYPE_STREAM
streamObj.Bytes = bytes
result.Stream = streamObj
}
if token != "endobj" {
return nil, errors.New("Expected next token to be: endobj, got: " + token)
}
// Reposition the file pointer to previous position
_, err = this.f.Seek(old_pos, 0)
if err != nil {
return nil, errors.Wrap(err, "Failed to set position of file")
}
return result, nil
} else {
return objSpec, nil
}
return &PdfValue{}, nil
}
// Find the xref offset (should be at the end of the PDF)
func (this *PdfReader) findXref() error {
var result int
var err error
var toRead int64
toRead = 1500
// If PDF is smaller than 1500 bytes, be sure to only read the number of bytes that are in the file
fileSize := this.nBytes
if fileSize < toRead {
toRead = fileSize
}
// 0 means relative to the origin of the file,
// 1 means relative to the current offset,
// and 2 means relative to the end.
whence := 2
// Perform seek operation
_, err = this.f.Seek(-toRead, whence)
if err != nil {
return errors.Wrap(err, "Failed to set position of file")
}
// Create new bufio.Reader
r := bufio.NewReader(this.f)
for {
// Read all tokens until "startxref" is found
token, err := this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if token == "startxref" {
token, err = this.readToken(r)
// Probably EOF before finding startxref
if err != nil {
return errors.Wrap(err, "Failed to find startxref token")
}
// Convert line (string) into int
result, err = strconv.Atoi(token)
if err != nil {
return errors.Wrap(err, "Failed to convert xref position into integer: "+token)
}
// Successfully read the xref position
this.xrefPos = result
break
}
// (token, err) == ("", nil) indicates EOF reached before finding "startxref"
if token == "" {
return errors.New("Failed to find startxref token")
}
}
// Rewind file pointer
whence = 0
_, err = this.f.Seek(0, whence)
if err != nil {
return errors.Wrap(err, "Failed to set position of file")
}
this.xrefPos = result
return nil
}
// Read and parse the xref table
func (this *PdfReader) readXref() error {
var err error
// Create new bufio.Reader
r := bufio.NewReader(this.f)
// Set file pointer to xref start
_, err = this.f.Seek(int64(this.xrefPos), 0)
if err != nil {
return errors.Wrap(err, "Failed to set position of file")
}
// Xref should start with 'xref'
t, err := this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if t != "xref" {
// Maybe this is an XRef stream ...
v, err := this.readValue(r, t)
if err != nil {
return errors.Wrap(err, "Failed to read XRef stream")
}
if v.Type == PDF_TYPE_OBJDEC {
// Read next token
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
// Read actual object value
v, err := this.readValue(r, t)
if err != nil {
return errors.Wrap(err, "Failed to read value for token: "+t)
}
// If /Type is set, check to see if it is XRef
if _, ok := v.Dictionary["/Type"]; ok {
if v.Dictionary["/Type"].Token == "/XRef" {
// Continue reading xref stream data now that it is confirmed that it is an xref stream
// Check for /DecodeParms
paethDecode := false
if _, ok := v.Dictionary["/DecodeParms"]; ok {
columns := 0
predictor := 0
if _, ok2 := v.Dictionary["/DecodeParms"].Dictionary["/Columns"]; ok2 {
columns = v.Dictionary["/DecodeParms"].Dictionary["/Columns"].Int
}
if _, ok2 := v.Dictionary["/DecodeParms"].Dictionary["/Predictor"]; ok2 {
predictor = v.Dictionary["/DecodeParms"].Dictionary["/Predictor"].Int
}
if columns > 4 || predictor > 12 {
return errors.New("Unsupported /DecodeParms - only tested with /Columns <= 4 and /Predictor <= 12")
}
paethDecode = true
}
/*
// Check to make sure field size is [1 2 1] - not yet tested with other field sizes
if v.Dictionary["/W"].Array[0].Int != 1 || v.Dictionary["/W"].Array[1].Int > 4 || v.Dictionary["/W"].Array[2].Int != 1 {
return errors.New(fmt.Sprintf("Unsupported field sizes in cross-reference stream dictionary: /W [%d %d %d]",
v.Dictionary["/W"].Array[0].Int,
v.Dictionary["/W"].Array[1].Int,
v.Dictionary["/W"].Array[2].Int))
}
*/
index := make([]int, 2)
// If /Index is not set, this is an error
if _, ok := v.Dictionary["/Index"]; ok {
if len(v.Dictionary["/Index"].Array) < 2 {
return errors.Wrap(err, "Index array does not contain 2 elements")
}
index[0] = v.Dictionary["/Index"].Array[0].Int
index[1] = v.Dictionary["/Index"].Array[1].Int
} else {
index[0] = 0
}
prevXref := 0
// Check for previous xref stream
if _, ok := v.Dictionary["/Prev"]; ok {
prevXref = v.Dictionary["/Prev"].Int
}
// Set root object
if _, ok := v.Dictionary["/Root"]; ok {
// Just set the whole dictionary with /Root key to keep compatibiltiy with existing code
this.trailer = v
} else {
// Don't return an error here. The trailer could be in another XRef stream.
//return errors.New("Did not set root object")
}
startObject := index[0]
err = this.skipWhitespace(r)
if err != nil {
return errors.Wrap(err, "Failed to skip whitespace")
}
// Get stream length dictionary
lengthDict := v.Dictionary["/Length"]
// Get number of bytes of stream
length := lengthDict.Int
// If lengthDict is an object reference, resolve the object and set length
if lengthDict.Type == PDF_TYPE_OBJREF {
lengthDict, err = this.resolveObject(lengthDict)
if err != nil {
return errors.Wrap(err, "Failed to resolve length object of stream")
}
// Set length to resolved object value
length = lengthDict.Value.Int
}
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if t != "stream" {
return errors.New("Expected next token to be: stream, got: " + t)
}
err = this.skipWhitespace(r)
if err != nil {
return errors.Wrap(err, "Failed to skip whitespace")
}
// Read length bytes
data := make([]byte, length)
// Cannot use reader.Read() because that may not read all the bytes
_, err := io.ReadFull(r, data)
if err != nil {
return errors.Wrap(err, "Failed to read bytes from buffer")
}
// Look for endstream token
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if t != "endstream" {
return errors.New("Expected next token to be: endstream, got: " + t)
}
// Look for endobj token
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if t != "endobj" {
return errors.New("Expected next token to be: endobj, got: " + t)
}
// Now decode zlib data
b := bytes.NewReader(data)
z, err := zlib.NewReader(b)
if err != nil {
return errors.Wrap(err, "zlib.NewReader error")
}
defer z.Close()
p, err := io.ReadAll(z)
if err != nil {
return errors.Wrap(err, "ioutil.ReadAll error")
}
objPos := 0
objGen := 0
i := startObject
// Decode result with paeth algorithm
var result []byte
b = bytes.NewReader(p)
firstFieldSize := v.Dictionary["/W"].Array[0].Int
middleFieldSize := v.Dictionary["/W"].Array[1].Int
lastFieldSize := v.Dictionary["/W"].Array[2].Int
fieldSize := firstFieldSize + middleFieldSize + lastFieldSize
if paethDecode {
fieldSize++
}
prevRow := make([]byte, fieldSize)
for {
result = make([]byte, fieldSize)
_, err := io.ReadFull(b, result)
if err != nil {
if err == io.EOF {
break
} else {
return errors.Wrap(err, "io.ReadFull error")
}
}
if paethDecode {
filterPaeth(result, prevRow, fieldSize)
copy(prevRow, result)
}
objectData := make([]byte, fieldSize)
if paethDecode {
copy(objectData, result[1:fieldSize])
} else {
copy(objectData, result[0:fieldSize])
}
if objectData[0] == 1 {
// Regular objects
b := make([]byte, 4)
copy(b[4-middleFieldSize:], objectData[1:1+middleFieldSize])
objPos = int(binary.BigEndian.Uint32(b))
objGen = int(objectData[firstFieldSize+middleFieldSize])
// Append map[int]int
this.xref[i] = make(map[int]int, 1)
// Set object id, generation, and position
this.xref[i][objGen] = objPos
} else if objectData[0] == 2 {
// Compressed objects
b := make([]byte, 4)
copy(b[4-middleFieldSize:], objectData[1:1+middleFieldSize])
objId := int(binary.BigEndian.Uint32(b))
objIdx := int(objectData[firstFieldSize+middleFieldSize])
// object id (i) is located in StmObj (objId) at index (objIdx)
this.xrefStream[i] = [2]int{objId, objIdx}
}
i++
}
// Check for previous xref stream
if prevXref > 0 {
// Set xrefPos to /Prev xref
this.xrefPos = prevXref
// Read preivous xref
xrefErr := this.readXref()
if xrefErr != nil {
return errors.Wrap(xrefErr, "Failed to read prev xref")
}
}
}
}
return nil
}
return errors.New("Expected xref to start with 'xref'. Got: " + t)
}
for {
// Next value will be the starting object id (usually 0, but not always) or the trailer
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
// Check for trailer
if t == "trailer" {
break
}
// Convert token to int
startObject, err := strconv.Atoi(t)
if err != nil {
return errors.Wrap(err, "Failed to convert start object to integer: "+t)
}
// Determine how many objects there are
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
// Convert token to int
numObject, err := strconv.Atoi(t)
if err != nil {
return errors.Wrap(err, "Failed to convert num object to integer: "+t)
}
// For all objects in xref, read object position, object generation, and status (free or new)
for i := startObject; i < startObject+numObject; i++ {
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
// Get object position as int
objPos, err := strconv.Atoi(t)
if err != nil {
return errors.Wrap(err, "Failed to convert object position to integer: "+t)
}
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
// Get object generation as int
objGen, err := strconv.Atoi(t)
if err != nil {
return errors.Wrap(err, "Failed to convert object generation to integer: "+t)
}
// Get object status (free or new)
objStatus, err := this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
if objStatus != "f" && objStatus != "n" {
return errors.New("Expected objStatus to be 'n' or 'f', got: " + objStatus)
}
// Append map[int]int
this.xref[i] = make(map[int]int, 1)
// Set object id, generation, and position
this.xref[i][objGen] = objPos
}
}
// Read trailer dictionary
t, err = this.readToken(r)
if err != nil {
return errors.Wrap(err, "Failed to read token")
}
trailer, err := this.readValue(r, t)
if err != nil {
return errors.Wrap(err, "Failed to read value for token: "+t)
}
// If /Root is set, then set trailer object so that /Root can be read later
if _, ok := trailer.Dictionary["/Root"]; ok {
this.trailer = trailer
}
// If a /Prev xref trailer is specified, parse that
if tr, ok := trailer.Dictionary["/Prev"]; ok {
// Resolve parent xref table
this.xrefPos = tr.Int
return this.readXref()
}
return nil
}
// Read root (catalog object)
func (this *PdfReader) readRoot() error {
var err error
rootObjSpec := this.trailer.Dictionary["/Root"]
// Read root (catalog)
this.catalog, err = this.resolveObject(rootObjSpec)
if err != nil {
return errors.Wrap(err, "Failed to resolve root object")
}
return nil
}
// Read kids (pages inside a page tree)
func (this *PdfReader) readKids(kids *PdfValue, r int) error {
// Loop through pages and add to result
for i := 0; i < len(kids.Array); i++ {
page, err := this.resolveObject(kids.Array[i])
if err != nil {
return errors.Wrap(err, "Failed to resolve page/pages object")
}
objType := page.Value.Dictionary["/Type"].Token
if objType == "/Page" {
// Set page and increment curPage
this.pages[this.curPage] = page
this.curPage++
} else if objType == "/Pages" {
// Resolve kids
subKids, err := this.resolveObject(page.Value.Dictionary["/Kids"])
if err != nil {
return errors.Wrap(err, "Failed to resolve kids")
}
// Recurse into page tree
err = this.readKids(subKids, r+1)
if err != nil {
return errors.Wrap(err, "Failed to read kids")
}
} else {
return errors.Wrap(err, fmt.Sprintf("Unknown object type '%s'. Expected: /Pages or /Page", objType))
}
}
return nil
}
// Read all pages in PDF
func (this *PdfReader) readPages() error {
var err error
// resolve_pages_dict
pagesDict, err := this.resolveObject(this.catalog.Value.Dictionary["/Pages"])
if err != nil {
return errors.Wrap(err, "Failed to resolve pages object")
}
// This will normally return itself
kids, err := this.resolveObject(pagesDict.Value.Dictionary["/Kids"])
if err != nil {
return errors.Wrap(err, "Failed to resolve kids object")
}
// Get number of pages
pageCount, err := this.resolveObject(pagesDict.Value.Dictionary["/Count"])
if err != nil {
return errors.Wrap(err, "Failed to get page count")
}
this.pageCount = pageCount.Int
// Allocate pages
this.pages = make([]*PdfValue, pageCount.Int)
// Read kids
err = this.readKids(kids, 0)
if err != nil {
return errors.Wrap(err, "Failed to read kids")
}
return nil
}
// Get references to page resources for a given page number
func (this *PdfReader) getPageResources(pageno int) (*PdfValue, error) {
var err error
// Check to make sure page exists in pages slice
if len(this.pages) < pageno {
return nil, errors.New(fmt.Sprintf("Page %d does not exist!!", pageno))
}
// Resolve page object
page, err := this.resolveObject(this.pages[pageno-1])
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve page object")
}
// Check to see if /Resources exists in Dictionary
if _, ok := page.Value.Dictionary["/Resources"]; ok {
// Resolve /Resources object
res, err := this.resolveObject(page.Value.Dictionary["/Resources"])
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve resources object")
}
// If type is PDF_TYPE_OBJECT, return its Value
if res.Type == PDF_TYPE_OBJECT {
return res.Value, nil
}
// Otherwise, returned the resolved object
return res, nil
} else {
// If /Resources does not exist, check to see if /Parent exists and return the /Resources
// from parent.
if _, ok := page.Value.Dictionary["/Parent"]; ok {
// Resolve parent object
parentPage, err := this.resolveObject(page.Value.Dictionary["/Parent"])
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve parent object")
}
// If /Parent object type is PDF_TYPE_OBJECT, look for /Resources in its Value
if parentPage.Type == PDF_TYPE_OBJECT {
if _, ok := parentPage.Value.Dictionary["/Resources"]; ok {
// Resolve /Resources object
res, err := this.resolveObject(parentPage.Value.Dictionary["/Resources"])
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve resources object from parent (object)")
}
// If type is PDF_TYPE_OBJECT, return its Value
if res.Type == PDF_TYPE_OBJECT {
return res.Value, nil
}
// Otherwise, returned the resolved object
return res, nil
}
// Return an empty PdfValue if /Resources is not found
return &PdfValue{}, nil
}
// If /Parent object type is PDF_TYPE_DICTIONARY, look for /Resources in its Dictionary
if parentPage.Type == PDF_TYPE_DICTIONARY {
if _, ok := parentPage.Dictionary["/Resources"]; ok {
// Resolve /Resources object
res, err := this.resolveObject(parentPage.Dictionary["/Resources"])
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve resources object from parent (dictionary)")
}
// If type is PDF_TYPE_OBJECT, return its Value
if res.Type == PDF_TYPE_OBJECT {
return res.Value, nil
}
// Otherwise, returned the resolved object
return res, nil
}
// Return an empty PdfValue if /Resources is not found
return &PdfValue{}, nil
}
}
}
// Return an empty PdfValue if we got here
// TODO: Improve error handling
return &PdfValue{}, nil
}
// Get page content and return a slice of PdfValue objects
func (this *PdfReader) getPageContent(objSpec *PdfValue) ([]*PdfValue, error) {
var err error
var content *PdfValue
// Allocate slice
contents := make([]*PdfValue, 0)
if objSpec.Type == PDF_TYPE_OBJREF {
// If objSpec is an object reference, resolve the object and append it to contents
content, err = this.resolveObject(objSpec)
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve object")
}
contents = append(contents, content)
} else if objSpec.Type == PDF_TYPE_ARRAY {
// If objSpec is an array, loop through the array and recursively get page content and append to contents
for i := 0; i < len(objSpec.Array); i++ {
tmpContents, err := this.getPageContent(objSpec.Array[i])
if err != nil {
return nil, errors.Wrap(err, "Failed to get page content")
}
for j := 0; j < len(tmpContents); j++ {
contents = append(contents, tmpContents[j])
}
}
}
return contents, nil
}
// Get content (i.e. PDF drawing instructions)
func (this *PdfReader) getContent(pageno int) (string, error) {
var err error
var contents []*PdfValue
var ascii85Err error
// Check to make sure page exists in pages slice
if len(this.pages) < pageno {
return "", errors.New(fmt.Sprintf("Page %d does not exist.", pageno))
}
// Get page
page := this.pages[pageno-1]
// FIXME: This could be slow, converting []byte to string and appending many times
buffer := ""
// Check to make sure /Contents exists in page dictionary
if _, ok := page.Value.Dictionary["/Contents"]; ok {
// Get an array of page content
contents, err = this.getPageContent(page.Value.Dictionary["/Contents"])
if err != nil {
return "", errors.Wrap(err, "Failed to get page content")
}
for i := 0; i < len(contents); i++ {
// If the pdf type is an object, check if its value is an array
// The actual content (Stream) is most likely here (This is the case for Canadapost shipping labels)
if contents[i].Type == PDF_TYPE_OBJECT {
if contents[i].Value != nil {
if contents[i].Value.Type == PDF_TYPE_ARRAY {
arrayContents, err := this.getPageContent(contents[i].Value)
if err != nil {
return "", errors.Wrap(err, "Failed to get page content from content array")
}
contents = append(contents, arrayContents...)
}
}
}
// If there is no stream, it will panic when trying to rebuild content stream
if contents[i].Stream == nil {
continue
}
// Decode content if one or more /Filter is specified.
// Most common filter is FlateDecode which can be uncompressed with zlib
tmpBuffer, err := this.rebuildContentStream(contents[i])
if err != nil {
switch err.(type) {
case *Ascii85DecodeError:
ascii85Err = err
default:
return "", errors.Wrap(err, "Failed to rebuild content stream")
}
}
// FIXME: This is probably slow
buffer += string(tmpBuffer)
}
}
if ascii85Err != nil {
return buffer, ascii85Err
}
return buffer, nil
}
// Rebuild content stream
// This will decode content if one or more /Filter (such as FlateDecode) is specified.
// If there are multiple filters, they will be decoded in the order in which they were specified.
func (this *PdfReader) rebuildContentStream(content *PdfValue) ([]byte, error) {
var err error
var tmpFilter *PdfValue
var asciiDecodeErr *Ascii85DecodeError
// Allocate slice of PdfValue
filters := make([]*PdfValue, 0)
// If content has a /Filter, append it to filters slice
if _, ok := content.Value.Dictionary["/Filter"]; ok {
filter := content.Value.Dictionary["/Filter"]
// If filter type is a reference, resolve it
if filter.Type == PDF_TYPE_OBJREF {
tmpFilter, err = this.resolveObject(filter)
if err != nil {
return nil, errors.Wrap(err, "Failed to resolve object")
}
filter = tmpFilter.Value
}
if filter.Type == PDF_TYPE_TOKEN {
// If filter type is a token (e.g. FlateDecode), appent it to filters slice
filters = append(filters, filter)
} else if filter.Type == PDF_TYPE_ARRAY {
// If filter type is an array, then there are multiple filters. Set filters variable to array value.
filters = filter.Array
}
}
// Set stream variable to content bytes
stream := content.Stream.Bytes
// Loop through filters and apply each filter to stream
for i := 0; i < len(filters); i++ {
switch filters[i].Token {
case "/FlateDecode":
// Uncompress zlib compressed data
var out bytes.Buffer
zlibReader, _ := zlib.NewReader(bytes.NewBuffer(stream))
defer zlibReader.Close()
io.Copy(&out, zlibReader)
// Set stream to uncompressed data
stream = out.Bytes()
case "/ASCII85Decode":
encoded := stream
// the -3 strips the end of data marker
decodedBytes, err := ioutil.ReadAll(ascii85.NewDecoder(bytes.NewBuffer(encoded[:len(encoded)-3])))
if err != nil {
if strings.Contains(err.Error(), "illegal ascii85 data") {
asciiDecodeErr = &Ascii85DecodeError{Message: err.Error()}
} else {
return nil, err
}
}
stream = decodedBytes
default:
return nil, errors.New("Unsupported filter: " + filters[i].Token)
}
}
if asciiDecodeErr != nil {
return stream, asciiDecodeErr
}
return stream, nil
}
func (this *PdfReader) getNumPages() (int, error) {
if this.pageCount == 0 {
return 0, errors.New("Page count is 0")
}
return this.pageCount, nil
}
func (this *PdfReader) getAllPageBoxes(k float64) (map[int]map[string]map[string]float64, error) {
var err error
// Allocate result with the number of available boxes
result := make(map[int]map[string]map[string]float64, len(this.pages))
for i := 1; i <= len(this.pages); i++ {
result[i], err = this.getPageBoxes(i, k)
if result[i] == nil {
return nil, errors.Wrap(err, "Unable to get page box")
}
}
return result, nil
}
// Get all page box data
func (this *PdfReader) getPageBoxes(pageno int, k float64) (map[string]map[string]float64, error) {
var err error
// Allocate result with the number of available boxes
result := make(map[string]map[string]float64, len(this.availableBoxes))
// Check to make sure page exists in pages slice
if len(this.pages) < pageno {
return nil, errors.New(fmt.Sprintf("Page %d does not exist?", pageno))
}
// Resolve page object
page, err := this.resolveObject(this.pages[pageno-1])
if err != nil {
return nil, errors.New("Failed to resolve page object")
}
// Loop through available boxes and add to result
for i := 0; i < len(this.availableBoxes); i++ {
box, err := this.getPageBox(page, this.availableBoxes[i], k)
if err != nil {
return nil, errors.New("Failed to get page box")
}
result[this.availableBoxes[i]] = box
}
return result, nil
}
// Get a specific page box value (e.g. MediaBox) and return its values
func (this *PdfReader) getPageBox(page *PdfValue, box_index string, k float64) (map[string]float64, error) {
var err error
var tmpBox *PdfValue
// Allocate 8 fields in result
result := make(map[string]float64, 8)
// Check to make sure box_index (e.g. MediaBox) exists in page dictionary
if _, ok := page.Value.Dictionary[box_index]; ok {
box := page.Value.Dictionary[box_index]
// If the box type is a reference, resolve it
if box.Type == PDF_TYPE_OBJREF {
tmpBox, err = this.resolveObject(box)
if err != nil {
return nil, errors.New("Failed to resolve object")
}
box = tmpBox.Value
}
if box.Type == PDF_TYPE_ARRAY {
// If the box type is an array, calculate scaled value based on k
result["x"] = box.Array[0].Real / k
result["y"] = box.Array[1].Real / k
result["w"] = math.Abs(box.Array[0].Real-box.Array[2].Real) / k
result["h"] = math.Abs(box.Array[1].Real-box.Array[3].Real) / k
result["llx"] = math.Min(box.Array[0].Real, box.Array[2].Real)
result["lly"] = math.Min(box.Array[1].Real, box.Array[3].Real)
result["urx"] = math.Max(box.Array[0].Real, box.Array[2].Real)
result["ury"] = math.Max(box.Array[1].Real, box.Array[3].Real)
} else {
// TODO: Improve error handling
return nil, errors.New("Could not get page box")
}
} else if _, ok := page.Value.Dictionary["/Parent"]; ok {
parentObj, err := this.resolveObject(page.Value.Dictionary["/Parent"])
if err != nil {
return nil, errors.Wrap(err, "Could not resolve parent object")
}
// If the page box is inherited from /Parent, recursively return page box of parent
return this.getPageBox(parentObj, box_index, k)
}
return result, nil
}
// Get page rotation for a page number
func (this *PdfReader) GetPageRotation(pageno int) (*PdfValue, error) {
// Check to make sure page exists in pages slice
if len(this.pages) < pageno {
return nil, errors.New(fmt.Sprintf("Page %d does not exist!!!!", pageno))
}
return this._getPageRotation(this.pages[pageno-1])
}
// Get page rotation for a page object spec
func (this *PdfReader) _getPageRotation(page *PdfValue) (*PdfValue, error) {
var err error
// Resolve page object
page, err = this.resolveObject(page)
if err != nil {
return nil, errors.New("Failed to resolve page object")
}
// Check to make sure /Rotate exists in page dictionary
if _, ok := page.Value.Dictionary["/Rotate"]; ok {
res, err := this.resolveObject(page.Value.Dictionary["/Rotate"])
if err != nil {
return nil, errors.New("Failed to resolve rotate object")
}
// If the type is PDF_TYPE_OBJECT, return its value
if res.Type == PDF_TYPE_OBJECT {
return res.Value, nil
}
// Otherwise, return the object
return res, nil
} else {
// Check to see if parent has a rotation
if _, ok := page.Value.Dictionary["/Parent"]; ok {
// Recursively return /Parent page rotation
res, err := this._getPageRotation(page.Value.Dictionary["/Parent"])
if err != nil {
return nil, errors.Wrap(err, "Failed to get page rotation for parent")
}
// If the type is PDF_TYPE_OBJECT, return its value
if res.Type == PDF_TYPE_OBJECT {
return res.Value, nil
}
// Otherwise, return the object
return res, nil
}
}
return &PdfValue{Int: 0}, nil
}
func (this *PdfReader) read() error {
// Only run once
if !this.alreadyRead {
var err error
// Find xref position
err = this.findXref()
if err != nil {
return errors.Wrap(err, "Failed to find xref position")
}
// Parse xref table
err = this.readXref()
if err != nil {
return errors.Wrap(err, "Failed to read xref table")
}
// Read catalog
err = this.readRoot()
if err != nil {
return errors.Wrap(err, "Failed to read root")
}
// Read pages
err = this.readPages()
if err != nil {
return errors.Wrap(err, "Failed to to read pages")
}
// Now that this has been read, do not read again
this.alreadyRead = true
}
return nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/jiecang-tech/gofpdi.git
git@gitee.com:jiecang-tech/gofpdi.git
jiecang-tech
gofpdi
gofpdi
v1.2.3

搜索帮助