189 lines
5.3 KiB
Go
189 lines
5.3 KiB
Go
package excelize
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"encoding/binary"
|
|
"math"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func BenchmarkWrite(b *testing.B) {
|
|
const s = "This is test data"
|
|
for i := 0; i < b.N; i++ {
|
|
f := NewFile()
|
|
for row := 1; row <= 10000; row++ {
|
|
for col := 1; col <= 20; col++ {
|
|
val, err := CoordinatesToCellName(col, row)
|
|
if err != nil {
|
|
b.Error(err)
|
|
}
|
|
if err := f.SetCellValue("Sheet1", val, s); err != nil {
|
|
b.Error(err)
|
|
}
|
|
}
|
|
}
|
|
// Save spreadsheet by the given path.
|
|
err := f.SaveAs("./test.xlsx")
|
|
if err != nil {
|
|
b.Error(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestWriteTo(t *testing.T) {
|
|
// Test WriteToBuffer err
|
|
{
|
|
f, buf := File{Pkg: sync.Map{}}, bytes.Buffer{}
|
|
f.Pkg.Store("/d/", []byte("s"))
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.EqualError(t, err, "zip: write to directory")
|
|
f.Pkg.Delete("/d/")
|
|
}
|
|
// Test file path overflow
|
|
{
|
|
f, buf := File{Pkg: sync.Map{}}, bytes.Buffer{}
|
|
const maxUint16 = 1<<16 - 1
|
|
f.Pkg.Store(strings.Repeat("s", maxUint16+1), nil)
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.EqualError(t, err, "zip: FileHeader.Name too long")
|
|
}
|
|
// Test StreamsWriter err
|
|
{
|
|
f, buf := File{Pkg: sync.Map{}}, bytes.Buffer{}
|
|
f.Pkg.Store("s", nil)
|
|
f.streams = make(map[string]*StreamWriter)
|
|
file, _ := os.Open("123")
|
|
f.streams["s"] = &StreamWriter{rawData: bufferedWriter{tmp: file}}
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.Nil(t, err)
|
|
}
|
|
// Test write with temporary file
|
|
{
|
|
f, buf := File{tempFiles: sync.Map{}}, bytes.Buffer{}
|
|
const maxUint16 = 1<<16 - 1
|
|
f.tempFiles.Store("s", "")
|
|
f.tempFiles.Store(strings.Repeat("s", maxUint16+1), "")
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.EqualError(t, err, "zip: FileHeader.Name too long")
|
|
}
|
|
// Test write with unsupported workbook file format
|
|
{
|
|
f, buf := File{Pkg: sync.Map{}}, bytes.Buffer{}
|
|
f.Pkg.Store("/d", []byte("s"))
|
|
f.Path = "Book1.xls"
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.EqualError(t, err, ErrWorkbookFileFormat.Error())
|
|
}
|
|
// Test write with unsupported charset content types.
|
|
{
|
|
f, buf := NewFile(), bytes.Buffer{}
|
|
f.ContentTypes, f.Path = nil, filepath.Join("test", "TestWriteTo.xlsx")
|
|
f.Pkg.Store(defaultXMLPathContentTypes, MacintoshCyrillicCharset)
|
|
_, err := f.WriteTo(bufio.NewWriter(&buf))
|
|
assert.EqualError(t, err, "XML syntax error on line 1: invalid UTF-8")
|
|
}
|
|
}
|
|
|
|
func TestClose(t *testing.T) {
|
|
f := NewFile()
|
|
f.tempFiles.Store("/d/", "/d/")
|
|
require.Error(t, f.Close())
|
|
}
|
|
|
|
func TestZip64(t *testing.T) {
|
|
f := NewFile()
|
|
_, err := f.NewSheet("Sheet2")
|
|
assert.NoError(t, err)
|
|
sw, err := f.NewStreamWriter("Sheet1")
|
|
assert.NoError(t, err)
|
|
for r := range 131 {
|
|
rowData := make([]interface{}, 1000)
|
|
for c := range 1000 {
|
|
rowData[c] = strings.Repeat("c", TotalCellChars)
|
|
}
|
|
cell, err := CoordinatesToCellName(1, r+1)
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, sw.SetRow(cell, rowData))
|
|
}
|
|
assert.NoError(t, sw.Flush())
|
|
assert.NoError(t, f.SaveAs(filepath.Join("test", "TestZip64.xlsx")))
|
|
assert.NoError(t, f.Close())
|
|
|
|
// Test with filename length overflow
|
|
f = NewFile()
|
|
f.zip64Entries = append(f.zip64Entries, defaultXMLPathSharedStrings)
|
|
buf := new(bytes.Buffer)
|
|
buf.Write([]byte{0x50, 0x4b, 0x03, 0x04})
|
|
buf.Write(make([]byte, 20))
|
|
assert.NoError(t, f.writeZip64LFH(buf))
|
|
|
|
// Test with file header less than the required 30 for the fixed header part
|
|
f = NewFile()
|
|
f.zip64Entries = append(f.zip64Entries, defaultXMLPathSharedStrings)
|
|
buf.Reset()
|
|
buf.Write([]byte{0x50, 0x4b, 0x03, 0x04})
|
|
buf.Write(make([]byte, 22))
|
|
binary.Write(buf, binary.LittleEndian, uint16(10))
|
|
buf.Write(make([]byte, 2))
|
|
buf.WriteString("test")
|
|
assert.NoError(t, f.writeZip64LFH(buf))
|
|
|
|
t.Run("for_save_zip64_with_in_memory_file_over_4GB", func(t *testing.T) {
|
|
// Test save workbook in ZIP64 format with in memory file with size over 4GB.
|
|
f := NewFile()
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
f.Pkg.Store("xl/worksheets/sheet1.xml", make([]byte, math.MaxUint32+1))
|
|
_, err := f.WriteToBuffer()
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, f.Close())
|
|
})
|
|
|
|
t.Run("for_save_zip64_with_in_temporary_file_over_4GB", func(t *testing.T) {
|
|
// Test save workbook in ZIP64 format with temporary file with size over 4GB.
|
|
if os.Getenv("GITHUB_ACTIONS") == "true" {
|
|
t.Skip()
|
|
}
|
|
f := NewFile()
|
|
f.Pkg.Delete("xl/worksheets/sheet1.xml")
|
|
f.Sheet.Delete("xl/worksheets/sheet1.xml")
|
|
tmp, err := os.CreateTemp(os.TempDir(), "excelize-")
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, tmp.Truncate(math.MaxUint32+1))
|
|
f.tempFiles.Store("xl/worksheets/sheet1.xml", tmp.Name())
|
|
assert.NoError(t, tmp.Close())
|
|
_, err = f.WriteToBuffer()
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, f.Close())
|
|
})
|
|
}
|
|
|
|
func TestRemoveTempFiles(t *testing.T) {
|
|
tmp, err := os.CreateTemp("", "excelize-*")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
tmpName := tmp.Name()
|
|
tmp.Close()
|
|
f := NewFile()
|
|
// fill the tempFiles map with non-existing (erroring on Remove) "files"
|
|
for i := 0; i < 1000; i++ {
|
|
f.tempFiles.Store(strconv.Itoa(i), "/hopefully not existing")
|
|
}
|
|
f.tempFiles.Store("existing", tmpName)
|
|
|
|
require.Error(t, f.Close())
|
|
if _, err := os.Stat(tmpName); err == nil {
|
|
t.Errorf("temp file %q still exist", tmpName)
|
|
os.Remove(tmpName)
|
|
}
|
|
}
|