forked from lukasjarosch/go-docx
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdocument.go
435 lines (380 loc) · 12.5 KB
/
document.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
package docx
import (
"archive/zip"
"bytes"
"errors"
"fmt"
"io"
"io/ioutil"
"log"
"os"
"path/filepath"
"regexp"
"strings"
"golang.org/x/net/html"
)
const (
// DocumentXml is the relative path where the actual document content resides inside the docx-archive.
DocumentXml = "word/document.xml"
)
var (
// HeaderPathRegex matches all header files inside the docx-archive.
HeaderPathRegex = regexp.MustCompile(`word/header[0-9]*.xml`)
// FooterPathRegex matches all footer files inside the docx-archive.
FooterPathRegex = regexp.MustCompile(`word/footer[0-9]*.xml`)
)
// Document exposes the main API of the library. It represents the actual docx document which is going to be modified.
// Although a 'docx' document actually consists of multiple xml files, that fact is not exposed via the Document API.
// All actions on the Document propagate through the files of the docx-zip-archive.
type Document struct {
path string
docxFile *os.File
zipFile *zip.Reader
// all files from the zip archive which we're interested in
files FileMap
// paths to all header files inside the zip archive
headerFiles []string
// paths to all footer files inside the zip archive
footerFiles []string
// The document contains multiple files which eventually need a parser each.
// The map key is the file path inside the document to which the parser belongs.
runParsers map[string]*RunParser
filePlaceholders map[string][]*Placeholder
fileReplacers map[string]*Replacer
}
// Open will open and parse the file pointed to by path.
// The file must be a valid docx file or an error is returned.
func Open(path string) (*Document, error) {
fh, err := os.Open(path)
if err != nil {
return nil, fmt.Errorf("unable to open .docx docxFile: %s", err)
}
rc, err := zip.OpenReader(path)
if err != nil {
return nil, fmt.Errorf("unable to open zip reader: %s", err)
}
return newDocument(&rc.Reader, path, fh)
}
// OpenBytes allows to create a Document from a byte slice.
// It behaves just like Open().
//
// Note: In this case, the docxFile property will be nil!
func OpenBytes(b []byte) (*Document, error) {
rc, err := zip.NewReader(bytes.NewReader(b), int64(len(b)))
if err != nil {
return nil, fmt.Errorf("unable to open zip reader: %s", err)
}
return newDocument(rc, "", nil)
}
// newDocument will create a new document struct given the zipFile.
// The params 'path' and 'docxFile' may be empty/nil in case the document is created from a byte source directly.
//
// newDocument will parse the docx archive and ValidatePositions that at least a 'document.xml' exists.
// If 'word/document.xml' is missing, an error is returned since the docx cannot be correct.
// Then all files are parsed for their runs before returning the new document.
func newDocument(zipFile *zip.Reader, path string, docxFile *os.File) (*Document, error) {
doc := &Document{
docxFile: docxFile,
zipFile: zipFile,
path: path,
files: make(FileMap),
runParsers: make(map[string]*RunParser),
filePlaceholders: make(map[string][]*Placeholder),
fileReplacers: make(map[string]*Replacer),
}
ResetRunIdCounter()
ResetFragmentIdCounter()
if err := doc.parseArchive(); err != nil {
return nil, fmt.Errorf("error parsing document: %s", err)
}
// a valid docx document should really contain a document.xml :)
if _, exists := doc.files[DocumentXml]; !exists {
return nil, fmt.Errorf("invalid docx archive, %s is missing", DocumentXml)
}
// parse all files
for name, data := range doc.files {
// find all runs
doc.runParsers[name] = NewRunParser(data)
err := doc.runParsers[name].Execute()
if err != nil {
return nil, err
}
// parse placeholders and initialize replacers
placeholder, err := ParsePlaceholders(doc.runParsers[name].Runs(), data)
if err != nil {
return nil, err
}
doc.filePlaceholders[name] = placeholder
doc.fileReplacers[name] = NewReplacer(data, placeholder)
}
return doc, nil
}
// ReplaceAll will iterate over all files and perform the replacement according to the PlaceholderMap.
func (d *Document) ReplaceAll(placeholderMap PlaceholderMap) error {
for name := range d.files {
changedBytes, err := d.replace(placeholderMap, name)
if err != nil {
return err
}
err = d.SetFile(name, changedBytes)
if err != nil {
return err
}
}
return nil
}
// Replace will attempt to replace the given key with the value in every file.
func (d *Document) Replace(key, value string) error {
for name := range d.files {
changedBytes, err := d.replace(PlaceholderMap{key: value}, name)
if err != nil {
return err
}
err = d.SetFile(name, changedBytes)
if err != nil {
return err
}
}
return nil
}
// replace will create a parser on the given bytes, execute it and replace every placeholders found with the data
// from the placeholderMap.
func (d *Document) replace(placeholderMap PlaceholderMap, file string) ([]byte, error) {
if _, ok := d.runParsers[file]; !ok {
return nil, fmt.Errorf("no parser for file %s", file)
}
placeholderCount := d.countPlaceholders(file, placeholderMap)
placeholders := d.filePlaceholders[file]
replacer := d.fileReplacers[file]
for key, value := range placeholderMap {
err := replacer.Replace(key, fmt.Sprint(value))
if err != nil {
if errors.Is(err, ErrPlaceholderNotFound) {
continue
} else {
return nil, err
}
}
}
// ensure that all placeholders have been replaced
if placeholderCount != replacer.ReplaceCount {
return nil, fmt.Errorf("not all placeholders were replaced, want=%d, have=%d", placeholderCount, replacer.ReplaceCount)
}
d.fileReplacers[file] = replacer
d.filePlaceholders[file] = placeholders
return replacer.Bytes(), nil
}
// Runs returns all runs from all parsed files.
func (d *Document) Runs() (runs []*Run) {
for _, parser := range d.runParsers {
runs = append(runs, parser.Runs()...)
}
return runs
}
// Placeholders returns all placeholders from the docx document.
func (d *Document) Placeholders() (placeholders []*Placeholder) {
for _, p := range d.filePlaceholders {
placeholders = append(placeholders, p...)
}
return placeholders
}
// countPlaceholders will return the total count of placeholders from the placeholderMap in the given data.
// Reoccurring placeholders are also counted multiple times.
func (d *Document) countPlaceholders(file string, placeholderMap PlaceholderMap) int {
data := d.GetFile(file)
plaintext := d.stripXmlTags(string(data))
var placeholderCount int
for key := range placeholderMap {
placeholder := AddPlaceholderDelimiter(key)
count := strings.Count(plaintext, placeholder)
if count > 0 {
placeholderCount += count
}
}
return placeholderCount
}
// stripXmlTags is a stdlib way of stripping out all xml tags using the html.Tokenizer.
// The returned string will be everything except the tags.
func (d *Document) stripXmlTags(data string) string {
var output string
tokenizer := html.NewTokenizer(strings.NewReader(data))
prevToken := tokenizer.Token()
loop:
for {
tok := tokenizer.Next()
switch {
case tok == html.ErrorToken:
break loop // End of the document, done
case tok == html.StartTagToken:
prevToken = tokenizer.Token()
case tok == html.TextToken:
if prevToken.Data == "script" {
continue
}
TxtContent := strings.TrimSpace(html.UnescapeString(string(tokenizer.Text())))
if len(TxtContent) > 0 {
output += TxtContent
}
}
}
return output
}
// GetFile returns the content of the given fileName if it exists.
func (d *Document) GetFile(fileName string) []byte {
if f, exists := d.files[fileName]; exists {
return f
}
return nil
}
// SetFile allows setting the file contents of the given file.
// The fileName must be known, otherwise an error is returned.
func (d *Document) SetFile(fileName string, fileBytes []byte) error {
if _, exists := d.files[fileName]; !exists {
return fmt.Errorf("unregistered file %s", fileName)
}
d.files[fileName] = fileBytes
return nil
}
// parseArchive will go through the docx zip archive and read them into the FileMap.
// Files inside the FileMap are those which can be modified by the lib.
// Currently not all files are read, only:
// - word/document.xml
// - word/header*.xml
// - word/footer*.xml
func (d *Document) parseArchive() error {
readZipFile := func(file *zip.File) []byte {
readCloser, err := file.Open()
if err != nil {
return nil
}
defer readCloser.Close()
fileBytes, err := ioutil.ReadAll(readCloser)
if err != nil {
return nil
}
return fileBytes
}
for _, file := range d.zipFile.File {
if file.Name == DocumentXml {
d.files[DocumentXml] = readZipFile(file)
}
if HeaderPathRegex.MatchString(file.Name) {
d.files[file.Name] = readZipFile(file)
d.headerFiles = append(d.headerFiles, file.Name)
}
if FooterPathRegex.MatchString(file.Name) {
d.files[file.Name] = readZipFile(file)
d.footerFiles = append(d.footerFiles, file.Name)
}
}
return nil
}
// WriteToFile will write the document to a new file.
// It is important to note that the target file cannot be the same as the path of this document.
// If the path is not yet created, the function will attempt to MkdirAll() before creating the file.
func (d *Document) WriteToFile(file string) error {
if file == d.path {
return fmt.Errorf("WriteToFile cannot write into the original docx archive while it'str open")
}
err := os.MkdirAll(filepath.Dir(file), 0755)
if err != nil {
return fmt.Errorf("unable to ensure path directories: %s", err)
}
target, err := os.Create(file)
if err != nil {
return err
}
defer target.Close()
return d.Write(target)
}
// Write is responsible for assembling a new .docx docxFile using the modified data as well as all remaining files.
// Docx files are basically zip archives with many XMLs included.
// Files which cannot be modified through this lib will just be read from the original docx and copied into the writer.
func (d *Document) Write(writer io.Writer) error {
zipWriter := zip.NewWriter(writer)
defer zipWriter.Close()
// writeModifiedFile will check if the given zipFile is a file which was modified and writes it.
// If the file is not one of the modified files, false is returned.
writeModifiedFile := func(writer io.Writer, zipFile *zip.File) (bool, error) {
isModified := d.isModifiedFile(zipFile.Name)
if !isModified {
return false, nil
}
if err := d.files.Write(writer, zipFile.Name); err != nil {
return false, fmt.Errorf("unable to writeFile %s: %s", zipFile.Name, err)
}
return true, nil
}
// write all files into the zip archive (docx-file)
for _, zipFile := range d.zipFile.File {
fw, err := zipWriter.Create(zipFile.Name)
if err != nil {
return fmt.Errorf("unable to create writer: %s", err)
}
// write all files which might've been modified by us
written, err := writeModifiedFile(fw, zipFile)
if err != nil {
return err
}
if written {
continue
}
// all files which we don't touch here (e.g. _rels.xml) are just copied from the original
readCloser, err := zipFile.Open()
if err != nil {
return fmt.Errorf("unable to open %s: %s", zipFile.Name, err)
}
_, err = fw.Write(readBytes(readCloser))
if err != nil {
return fmt.Errorf("unable to writeFile zipFile %s: %s", zipFile.Name, err)
}
err = readCloser.Close()
if err != nil {
return fmt.Errorf("unable to close reader for %s: %s", zipFile.Name, err)
}
}
return nil
}
// isModifiedFile will look through all modified files and check if the searchFileName exists
func (d *Document) isModifiedFile(searchFileName string) bool {
allFiles := append(d.headerFiles, d.footerFiles...)
allFiles = append(allFiles, DocumentXml)
for _, file := range allFiles {
if searchFileName == file {
return true
}
}
return false
}
// Close will close everything :)
func (d *Document) Close() {
if d.docxFile != nil {
err := d.docxFile.Close()
if err != nil {
log.Fatal(err)
}
}
}
// FileMap is just a convenience type for the map of fileName => fileBytes
type FileMap map[string][]byte
// Write will try to write the bytes from the map into the given writer.
func (fm FileMap) Write(writer io.Writer, filename string) error {
file, ok := fm[filename]
if !ok {
return fmt.Errorf("file not found %s", filename)
}
_, err := writer.Write(file)
if err != nil && err != io.EOF {
return fmt.Errorf("unable to writeFile '%s': %s", filename, err)
}
return nil
}
// readBytes reads an io.Reader into []byte and returns it.
func readBytes(stream io.Reader) []byte {
buf := new(bytes.Buffer)
n, err := buf.ReadFrom(stream)
if n == 0 || err != nil {
return buf.Bytes()
}
return buf.Bytes()
}