2017-09-03 08:17:57 -05:00
|
|
|
// Copyright 2017 Baliance. All rights reserved.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by the terms of the Affero GNU General
|
|
|
|
// Public License version 3.0 as published by the Free Software Foundation and
|
|
|
|
// appearing in the file LICENSE included in the packaging of this file. A
|
|
|
|
// commercial license can be purchased by contacting sales@baliance.com.
|
|
|
|
|
|
|
|
package spreadsheet
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/zip"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
|
2017-09-08 17:06:52 -05:00
|
|
|
"baliance.com/gooxml"
|
2017-09-03 08:17:57 -05:00
|
|
|
"baliance.com/gooxml/zippkg"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Read reads a workbook from an io.Reader(.xlsx).
|
|
|
|
func Read(r io.ReaderAt, size int64) (*Workbook, error) {
|
|
|
|
wb := New()
|
|
|
|
td, err := ioutil.TempDir("", "gooxml-xlsx")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
wb.TmpPath = td
|
|
|
|
|
|
|
|
zr, err := zip.NewReader(r, size)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("parsing zip: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
files := []*zip.File{}
|
|
|
|
files = append(files, zr.File...)
|
|
|
|
decMap := zippkg.DecodeMap{}
|
|
|
|
decMap.SetOnNewRelationshipFunc(wb.onNewRelationship)
|
|
|
|
// we should discover all contents by starting with these two files
|
2017-09-28 17:05:38 -05:00
|
|
|
decMap.AddTarget(gooxml.ContentTypesFilename, wb.ContentTypes.X(), "", 0)
|
|
|
|
decMap.AddTarget(gooxml.BaseRelsFilename, wb.Rels.X(), "", 0)
|
2017-09-08 21:25:01 -05:00
|
|
|
if err := decMap.Decode(files); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-09-03 08:17:57 -05:00
|
|
|
|
|
|
|
// etra files are things we don't handle yet, or files that happened to have
|
|
|
|
// been in the zip before. We just round-trip them.
|
|
|
|
for _, f := range files {
|
|
|
|
if f == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := wb.AddExtraFileFromZip(f); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return wb, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open opens and reads a workbook from a file (.xlsx).
|
|
|
|
func Open(filename string) (*Workbook, error) {
|
|
|
|
f, err := os.Open(filename)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error opening %s: %s", filename, err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
fi, err := os.Stat(filename)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error opening %s: %s", filename, err)
|
|
|
|
}
|
|
|
|
return Read(f, fi.Size())
|
|
|
|
}
|