unioffice/document/document_test.go
Gunnsteinn Hall 5335bf249b
License update (#426)
* update license and terms

* Fixes

* Create ACKNOWLEDGEMENTS.md

* Update ACKNOWLEDGEMENTS.md

* Revert go.mod changes and remove go1.11 tests
2020-08-08 01:01:05 +00:00

500 lines
15 KiB
Go

// Copyright 2017 FoxyUtils ehf. All rights reserved.
//
// Use of this software package and source code is governed by the terms of the
// UniDoc End User License Agreement (EULA) that is available at:
// https://unidoc.io/eula/
// A trial license code for evaluation can be obtained at https://unidoc.io.
package document_test
import (
"bytes"
"io/ioutil"
"os"
"testing"
"github.com/unidoc/unioffice/common"
"github.com/unidoc/unioffice/document"
"github.com/unidoc/unioffice/schema/soo/wml"
"github.com/unidoc/unioffice/testhelper"
)
func TestSimpleDoc(t *testing.T) {
doc := document.New()
testVersion := "00.8000"
doc.AppProperties.X().AppVersion = &testVersion
para := doc.AddParagraph()
run := para.AddRun()
run.AddText("foo")
got := bytes.Buffer{}
if err := doc.Validate(); err != nil {
t.Errorf("created an invalid document: %s", err)
}
doc.Save(&got)
testhelper.CompareGoldenZip(t, "simple-1.docx", got.Bytes())
}
func TestOpen(t *testing.T) {
wb, err := document.Open("testdata/simple-1.docx")
if err != nil {
t.Errorf("error opening document: %s", err)
}
got := bytes.Buffer{}
if err := wb.Validate(); err != nil {
t.Errorf("created an invalid document: %s", err)
}
wb.Save(&got)
testhelper.CompareZip(t, "simple-1.docx", got.Bytes(), true)
}
func TestOpenStrict(t *testing.T) {
strict, err := document.Open("testdata/strict.docx")
if err != nil {
t.Errorf("error opening document: %s", err)
}
gotStrict := bytes.Buffer{}
if err := strict.Validate(); err != nil {
t.Errorf("created an invalid document: %s", err)
}
strict.Save(&gotStrict)
ioutil.WriteFile("testdata/non-strict.docx", gotStrict.Bytes(), 0644)
// run test assuming that the doc is a valid non-strict doc
nonStrict, err := document.Open("testdata/non-strict.docx")
if err != nil {
t.Errorf("error opening document: %s", err)
}
gotNonStrict := bytes.Buffer{}
if err := nonStrict.Validate(); err != nil {
t.Errorf("created an invalid document: %s", err)
}
nonStrict.Save(&gotNonStrict)
testhelper.CompareZip(t, "non-strict.docx", gotNonStrict.Bytes(), true)
os.Remove("testdata/non-strict.docx")
}
func TestOpenHeaderFooter(t *testing.T) {
wb, err := document.Open("testdata/header-footer-multiple.docx")
if err != nil {
t.Errorf("error opening document: %s", err)
}
got := bytes.Buffer{}
if err := wb.Validate(); err != nil {
t.Errorf("created an invalid document: %s", err)
}
wb.Save(&got)
testhelper.CompareGoldenZip(t, "header-footer-multiple.docx", got.Bytes())
}
func TestAddParagraph(t *testing.T) {
doc := document.New()
if len(doc.Paragraphs()) != 0 {
t.Errorf("expected 0 paragraphs, got %d", len(doc.Paragraphs()))
}
doc.AddParagraph()
if len(doc.Paragraphs()) != 1 {
t.Errorf("expected 1 paragraphs, got %d", len(doc.Paragraphs()))
}
doc.AddParagraph()
if len(doc.Paragraphs()) != 2 {
t.Errorf("expected 2 paragraphs, got %d", len(doc.Paragraphs()))
}
}
func TestOpenWord2016(t *testing.T) {
doc, err := document.Open("../testdata/Office2016/Word-Windows.docx")
if err != nil {
t.Errorf("error opening Windows Word 2016 document: %s", err)
}
got := bytes.Buffer{}
if err := doc.Save(&got); err != nil {
t.Errorf("error saving W216 file: %s", err)
}
testhelper.CompareGoldenZipFilesOnly(t, "../../testdata/Office2016/Word-Windows.docx", got.Bytes())
}
func TestInsertParagraph(t *testing.T) {
doc := document.New()
if len(doc.Paragraphs()) != 0 {
t.Errorf("expected 0 paragraphs, got %d", len(doc.Paragraphs()))
}
p := doc.AddParagraph()
before := doc.InsertParagraphBefore(p)
after := doc.InsertParagraphAfter(p)
if len(doc.Paragraphs()) != 3 {
t.Errorf("expected 3 paragraphs, got %d", len(doc.Paragraphs()))
}
if doc.Paragraphs()[0].X() != before.X() {
t.Error("InsertParagraphBefore failed")
}
if doc.Paragraphs()[2].X() != after.X() {
t.Error("InsertParagraphAfter failed")
}
}
func TestInsertTable(t *testing.T) {
doc := document.New()
if len(doc.Paragraphs()) != 0 {
t.Errorf("expected 0 paragraphs, got %d", len(doc.Paragraphs()))
}
p1 := doc.AddParagraph()
p2 := doc.AddParagraph()
beforeP1 := doc.InsertTableBefore(p1)
afterP1 := doc.InsertTableAfter(p1)
beforeP2 := doc.InsertTableBefore(p2)
afterP2 := doc.InsertTableAfter(p2)
if len(doc.Tables()) != 4 {
t.Errorf("expected 4 tables, got %d", len(doc.Tables()))
}
if doc.Tables()[0].X() != beforeP1.X() {
t.Error("InsertTableBefore 1st paragraph failed")
}
if doc.Tables()[1].X() != afterP1.X() {
t.Error("InsertTableAfter 1st paragraph failed")
}
if doc.Tables()[2].X() != beforeP2.X() {
t.Error("InsertTableBefore 2nd paragraph failed")
}
if doc.Tables()[3].X() != afterP2.X() {
t.Error("InsertTableAfter 2nd paragraph failed")
}
}
func TestInsertRun(t *testing.T) {
doc := document.New()
if len(doc.Paragraphs()) != 0 {
t.Errorf("expected 0 paragraphs, got %d", len(doc.Paragraphs()))
}
p := doc.AddParagraph()
middle := p.AddRun()
before := p.InsertRunBefore(middle)
after := p.InsertRunAfter(middle)
middle.AddText("middle")
before.AddText("before")
after.AddText("after")
if len(p.Runs()) != 3 {
t.Errorf("expected 3 runs, got %d", len(p.Runs()))
}
if p.Runs()[0].X() != before.X() {
t.Error("InsertParagraphBefore failed")
}
if p.Runs()[2].X() != after.X() {
t.Error("InsertParagraphAfter failed")
}
p.RemoveRun(after)
if len(p.Runs()) != 2 {
t.Errorf("expected 2 runs, got %d", len(p.Runs()))
}
if p.Runs()[0].X() != before.X() {
t.Error("InsertParagraphBefore failed")
}
p.RemoveRun(before)
if len(p.Runs()) != 1 {
t.Errorf("expected 1 runs, got %d", len(p.Runs()))
}
if p.Runs()[0].X() != middle.X() {
t.Errorf("remove failed")
}
}
func TestInsertBookmarks(t *testing.T) {
doc := document.New()
if len(doc.Bookmarks()) != 0 {
t.Errorf("expected 0 bookmarks, got %d", len(doc.Bookmarks()))
}
p := doc.AddParagraph()
p.AddBookmark("bookmark1")
p.AddBookmark("bookmark2")
if len(doc.Bookmarks()) != 2 {
t.Errorf("expected 2 bookmarks, got %d", len(doc.Bookmarks()))
}
}
func TestDuplicateBookmarks(t *testing.T) {
doc := document.New()
if len(doc.Bookmarks()) != 0 {
t.Errorf("expected 0 bookmarks, got %d", len(doc.Bookmarks()))
}
p := doc.AddParagraph()
p.AddBookmark("bookmark1")
p.AddBookmark("bookmark1")
if len(doc.Bookmarks()) != 2 {
t.Errorf("expected 2 bookmarks, got %d", len(doc.Bookmarks()))
}
if err := doc.Validate(); err == nil {
t.Errorf("expected error due to duplicate bookmark names")
}
}
func TestHeaderAndFooterImages(t *testing.T) {
doc := document.New()
img1, err := common.ImageFromFile("testdata/gopher.png")
if err != nil {
t.Fatalf("unable to create image: %s", err)
}
img2, err := common.ImageFromFile("testdata/gophercolor.png")
if err != nil {
t.Fatalf("unable to create image: %s", err)
}
png3x3 := []byte{
0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a,
0x00, 0x00, 0x00, 0x0d, 0x49, 0x48, 0x44, 0x52,
0x00, 0x00, 0x00, 0x03, 0x00, 0x00, 0x00, 0x03,
0x08, 0x02, 0x00, 0x00, 0x00, 0xd9, 0x4a, 0x22,
0xe8, 0x00, 0x00, 0x00, 0x1e, 0x49, 0x44, 0x41,
0x54, 0x08, 0xd7, 0x63, 0xf8, 0xc5, 0x1e, 0xf8,
0x9d, 0xfd, 0xd7, 0x34, 0xf6, 0x5f, 0x0c, 0x10,
0x8a, 0x9d, 0xf7, 0x17, 0x03, 0x84, 0x62, 0xf7,
0xf9, 0x05, 0x00, 0xd2, 0x6f, 0x0d, 0x71, 0x26,
0x33, 0x2f, 0xe1, 0x00, 0x00, 0x00, 0x00, 0x49,
0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82,
}
img3, err := common.ImageFromBytes(png3x3)
if err != nil {
t.Fatalf("unable to create image: %s", err)
}
dir1, err := doc.AddImage(img1)
if err != nil {
t.Fatalf("unable to add image to doc: %s", err)
}
if dir1.RelID() != "rId4" {
t.Errorf("expected rId4 != %s", dir1.RelID())
}
dir2, err := doc.AddImage(img2)
if err != nil {
t.Fatalf("unable to add image to doc: %s", err)
}
if dir2.RelID() != "rId5" {
t.Errorf("expected rId5 != %s", dir2.RelID())
}
dir3, err := doc.AddImage(img3)
if err != nil {
t.Fatalf("unable to add image to doc: %s", err)
}
if dir3.RelID() != "rId6" {
t.Errorf("expected rId6 != %s", dir3.RelID())
}
hdr := doc.AddHeader()
ftr := doc.AddFooter()
hir1, err := hdr.AddImage(img1)
if err != nil {
t.Fatalf("unable to add image to header: %s", err)
}
if hir1.RelID() != "rId1" {
t.Errorf("expected rId1 != %s", hir1.RelID())
}
hir2, err := hdr.AddImage(img2)
if err != nil {
t.Fatalf("unable to add image to header: %s", err)
}
if hir2.RelID() != "rId2" {
t.Errorf("expected rId2 != %s", hir2.RelID())
}
hir3, err := hdr.AddImage(img3)
if err != nil {
t.Fatalf("unable to add image to header: %s", err)
}
if hir3.RelID() != "rId3" {
t.Errorf("expected rId3 != %s", hir3.RelID())
}
fir1, err := ftr.AddImage(img1)
if err != nil {
t.Fatalf("unable to add image to footer: %s", err)
}
if fir1.RelID() != "rId1" {
t.Errorf("expected rId1 != %s", fir1.RelID())
}
fir2, err := ftr.AddImage(img2)
if err != nil {
t.Fatalf("unable to add image to footer: %s", err)
}
if fir2.RelID() != "rId2" {
t.Errorf("expected rId2 != %s", fir2.RelID())
}
fir3, err := ftr.AddImage(img3)
if err != nil {
t.Fatalf("unable to add image to footer: %s", err)
}
if fir3.RelID() != "rId3" {
t.Errorf("expected rId3 != %s", fir3.RelID())
}
}
func TestIssue198(t *testing.T) {
// this tests the image fixes performed as part of issue 198
// where we were breaking jpg images
fn := "issue198.docx"
doc, err := document.Open("testdata/" + fn)
if err != nil {
t.Errorf("error reading %s: %s", fn, err)
return
}
got := bytes.Buffer{}
doc.Save(&got)
testhelper.CompareGoldenZip(t, fn+".golden", got.Bytes())
}
func TestGetTables(t *testing.T) {
doc := document.New()
table := doc.AddTable()
tables := doc.Tables()
if len(tables) != 1 {
t.Errorf("expected 1 table, got %d", len(tables))
return
}
if table != tables[0] {
t.Error("retrieved table != added table")
return
}
tbl := document.New().AddTable().X()
tc := table.AddRow().AddCell().X()
elts := wml.NewEG_BlockLevelElts()
tc.EG_BlockLevelElts = append(tc.EG_BlockLevelElts, elts)
c := wml.NewEG_ContentBlockContent()
elts.EG_ContentBlockContent = append(elts.EG_ContentBlockContent, c)
c.Tbl = append(c.Tbl, tbl)
tables = doc.Tables()
if len(tables) < 2 {
t.Errorf("nested table not enumerated. found %d, expected 2", len(tables))
}
}
func TestInsertParagraphInTable(t *testing.T) {
doc := document.New()
paraBeforeTable := doc.AddParagraph()
paraBeforeTable.AddRun().AddText("before table")
table := doc.InsertTableAfter(paraBeforeTable)
tablePara1 := table.AddRow().AddCell().AddParagraph()
tablePara1.AddRun().AddText("table paragraph 1")
paraAfterTable := doc.AddParagraph()
paraAfterTable.AddRun().AddText("after table")
tablePara2 := doc.InsertParagraphBefore(tablePara1)
tablePara2.AddRun().AddText("table paragraph before table paragraph 1")
tablePara3 := doc.InsertParagraphAfter(tablePara1)
tablePara3.AddRun().AddText("table paragraph after table paragraph 1")
bles := doc.X().Body.EG_BlockLevelElts
if len(bles) != 3 {
t.Errorf("expected 3 block level elements, got %d", len(bles))
}
if len(bles[0].EG_ContentBlockContent[0].P) != 1 {
t.Errorf("expected 1 paragraph in the first block level element, got %d", len(bles[0].EG_ContentBlockContent[0].P))
}
if len(bles[1].EG_ContentBlockContent[0].P) != 0 {
t.Errorf("expected no paragraphs in the second block level element, got %d", len(bles[1].EG_ContentBlockContent[0].P))
}
if len(bles[1].EG_ContentBlockContent[0].Tbl) != 1 {
t.Errorf("expected 1 table in the second block level element, got %d", len(bles[1].EG_ContentBlockContent[0].Tbl))
}
if len(bles[2].EG_ContentBlockContent[0].P) != 1 {
t.Errorf("expected 1 paragraph in the third block level element, got %d", len(bles[2].EG_ContentBlockContent[0].P))
}
cbc := bles[1].EG_ContentBlockContent[0].Tbl[0].EG_ContentRowContent[0].Tr[0].EG_ContentCellContent[0].Tc[0].EG_BlockLevelElts[0].EG_ContentBlockContent[0]
if len(cbc.P) != 3 {
t.Errorf("expected 3 paragraphs in the table, got %d", len(cbc.P))
}
expected := []string{"table paragraph before table paragraph 1", "table paragraph 1", "table paragraph after table paragraph 1"}
for i, p := range cbc.P {
got := p.EG_PContent[0].EG_ContentRunContent[0].R.EG_RunInnerContent[0].T.Content
if got != expected[i] {
t.Errorf("expected %s in %d table paragraph, got %s", expected[i], i, got)
}
}
}
func TestInsertTableInTable(t *testing.T) {
doc := document.New()
paraBeforeTable := doc.AddParagraph()
paraBeforeTable.AddRun().AddText("before table")
table := doc.InsertTableAfter(paraBeforeTable)
row := table.AddRow()
tablePara1 := row.AddCell().AddParagraph()
tablePara1.AddRun().AddText("table paragraph 1")
paraAfterTable := doc.AddParagraph()
paraAfterTable.AddRun().AddText("after table")
tableInTable1 := doc.InsertTableBefore(tablePara1)
ttPara1 := tableInTable1.AddRow().AddCell().AddParagraph()
ttPara1.AddRun().AddText("table inside table before paragraph 1")
tableInTable2 := doc.InsertTableAfter(tablePara1)
ttPara2 := tableInTable2.AddRow().AddCell().AddParagraph()
ttPara2.AddRun().AddText("table inside table after paragraph 1")
bles := doc.X().Body.EG_BlockLevelElts
if len(bles) != 3 {
t.Errorf("expected 3 block level elements, got %d", len(bles))
}
if len(bles[0].EG_ContentBlockContent[0].P) != 1 {
t.Errorf("expected 1 paragraph in the first block level element, got %d", len(bles[0].EG_ContentBlockContent[0].P))
}
if len(bles[1].EG_ContentBlockContent[0].P) != 0 {
t.Errorf("expected no paragraphs in the second block level element, got %d", len(bles[1].EG_ContentBlockContent[0].P))
}
if len(bles[1].EG_ContentBlockContent[0].Tbl) != 1 {
t.Errorf("expected 1 table in the second block level element, got %d", len(bles[1].EG_ContentBlockContent[0].Tbl))
}
if len(bles[2].EG_ContentBlockContent[0].P) != 1 {
t.Errorf("expected 1 paragraph in the third block level element, got %d", len(bles[2].EG_ContentBlockContent[0].P))
}
elts := bles[1].EG_ContentBlockContent[0].Tbl[0].EG_ContentRowContent[0].Tr[0].EG_ContentCellContent[0].Tc[0].EG_BlockLevelElts
cbc0 := elts[0].EG_ContentBlockContent[0]
if len(cbc0.Tbl) != 1 {
t.Errorf("expected 1 table as the first element of table, got %d", len(cbc0.Tbl))
}
cbc1 := elts[1].EG_ContentBlockContent[0]
if len(cbc1.P) != 1 {
t.Errorf("expected 1 paragraph as the second element of table, got %d", len(cbc1.P))
}
cbc2 := elts[2].EG_ContentBlockContent[0]
if len(cbc2.Tbl) != 1 {
t.Errorf("expected 1 table as the third element of table, got %d", len(cbc2.Tbl))
}
expected := "table inside table after paragraph 1"
got := cbc2.Tbl[0].EG_ContentRowContent[0].Tr[0].EG_ContentCellContent[0].Tc[0].EG_BlockLevelElts[0].EG_ContentBlockContent[0].P[0].EG_PContent[0].EG_ContentRunContent[0].R.EG_RunInnerContent[0].T.Content
if expected != got {
t.Errorf("expected %s in the second inner table paragraph, got %s", expected, got)
}
}