sq/drivers/xlsx/xlsx.go
Neil O'Toole 3f6157c4c4
Change logging library to slog (#175)
- Switch to slog logger.
2023-04-02 13:49:45 -06:00

316 lines
7.5 KiB
Go

// Package xlsx implements the sq driver for Microsoft Excel.
package xlsx
import (
"context"
"database/sql"
"io"
"github.com/neilotoole/sq/libsq/core/lg/lga"
"github.com/neilotoole/sq/libsq/core/lg/lgm"
"github.com/neilotoole/sq/libsq/core/lg"
"golang.org/x/exp/slog"
"github.com/tealeg/xlsx/v2"
"github.com/neilotoole/sq/libsq/core/cleanup"
"github.com/neilotoole/sq/libsq/core/errz"
"github.com/neilotoole/sq/libsq/core/options"
"github.com/neilotoole/sq/libsq/driver"
"github.com/neilotoole/sq/libsq/source"
)
const (
// Type is the sq source driver type for XLSX.
Type = source.Type("xlsx")
)
// Provider implements driver.Provider.
type Provider struct {
Log *slog.Logger
Files *source.Files
Scratcher driver.ScratchDatabaseOpener
}
// DriverFor implements driver.Provider.
func (p *Provider) DriverFor(typ source.Type) (driver.Driver, error) {
if typ != Type {
return nil, errz.Errorf("unsupported driver type {%s}", typ)
}
return &Driver{log: p.Log, scratcher: p.Scratcher, files: p.Files}, nil
}
var _ source.TypeDetectFunc = DetectXLSX
// DetectXLSX implements source.TypeDetectFunc, returning
// TypeXLSX and a score of 1.0 valid XLSX.
func DetectXLSX(ctx context.Context, openFn source.FileOpenFunc) (detected source.Type, score float32,
err error,
) {
log := lg.FromContext(ctx)
var r io.ReadCloser
r, err = openFn()
if err != nil {
return source.TypeNone, 0, errz.Err(err)
}
defer lg.WarnIfCloseError(log, lgm.CloseFileReader, r)
data, err := io.ReadAll(r)
if err != nil {
return source.TypeNone, 0, errz.Err(err)
}
// We don't need to read all rows, one will do.
const rowLimit = 1
_, err = xlsx.OpenBinaryWithRowLimit(data, rowLimit)
if err != nil {
return source.TypeNone, 0, nil
}
return Type, 1.0, nil
}
// Driver implements driver.Driver.
type Driver struct {
log *slog.Logger
scratcher driver.ScratchDatabaseOpener
files *source.Files
}
// DriverMetadata implements driver.Driver.
func (d *Driver) DriverMetadata() driver.Metadata {
return driver.Metadata{
Type: Type,
Description: "Microsoft Excel XLSX",
Doc: "https://en.wikipedia.org/wiki/Microsoft_Excel",
}
}
// Open implements driver.Driver.
func (d *Driver) Open(ctx context.Context, src *source.Source) (driver.Database, error) {
r, err := d.files.Open(src)
if err != nil {
return nil, err
}
defer lg.WarnIfCloseError(d.log, lgm.CloseFileReader, r)
b, err := io.ReadAll(r)
if err != nil {
return nil, errz.Err(err)
}
xlFile, err := xlsx.OpenBinary(b)
if err != nil {
return nil, err
}
scratchDB, err := d.scratcher.OpenScratch(ctx, src.Handle)
if err != nil {
return nil, err
}
clnup := cleanup.New()
clnup.AddE(scratchDB.Close)
err = xlsxToScratch(ctx, src, xlFile, scratchDB)
if err != nil {
lg.WarnIfError(d.log, lgm.CloseDB, clnup.Run())
return nil, err
}
return &database{log: d.log, src: src, impl: scratchDB, files: d.files, clnup: clnup}, nil
}
// Truncate implements driver.Driver.
func (d *Driver) Truncate(_ context.Context, src *source.Source, _ string, _ bool) (affected int64, err error) {
// TODO: WE could actually implement Truncate for xlsx.
// It would just mean deleting the rows from a sheet, and then
// saving the sheet.
return 0, errz.Errorf("source type {%s} (%s) doesn't support dropping tables", Type, src.Handle)
}
// ValidateSource implements driver.Driver.
func (d *Driver) ValidateSource(src *source.Source) (*source.Source, error) {
d.log.Debug("Validating source: {%s}", src.RedactedLocation())
if src.Type != Type {
return nil, errz.Errorf("expected source type {%s} but got {%s}", Type, src.Type)
}
return src, nil
}
// Ping implements driver.Driver.
func (d *Driver) Ping(_ context.Context, src *source.Source) (err error) {
r, err := d.files.Open(src)
if err != nil {
return err
}
defer lg.WarnIfCloseError(d.log, lgm.CloseFileReader, r)
b, err := io.ReadAll(r)
if err != nil {
return errz.Err(err)
}
_, err = xlsx.OpenBinaryWithRowLimit(b, 1)
if err != nil {
return errz.Err(err)
}
return nil
}
// database implements driver.Database.
type database struct {
log *slog.Logger
src *source.Source
files *source.Files
impl driver.Database
clnup *cleanup.Cleanup
}
// DB implements driver.Database.
func (d *database) DB() *sql.DB {
return d.impl.DB()
}
// SQLDriver implements driver.Database.
func (d *database) SQLDriver() driver.SQLDriver {
return d.impl.SQLDriver()
}
// Source implements driver.Database.
func (d *database) Source() *source.Source {
return d.src
}
// SourceMetadata implements driver.Database.
//
// TODO: the implementation of SourceMetadata is out
// of sync with the way we import data. For example, empty
// rows are filtered out during import, and empty columns
// are discarded. Thus SourceMetadata needs an overhaul to
// bring its reporting into line with import.
func (d *database) SourceMetadata(_ context.Context) (*source.Metadata, error) {
meta := &source.Metadata{Handle: d.src.Handle}
var err error
meta.Size, err = d.files.Size(d.src)
if err != nil {
return nil, err
}
meta.Name, err = source.LocationFileName(d.src)
if err != nil {
return nil, err
}
meta.FQName = meta.Name
meta.Location = d.src.Location
meta.SourceType = Type
b, err := d.files.ReadAll(d.src)
if err != nil {
return nil, errz.Err(err)
}
xlFile, err := xlsx.OpenBinary(b)
if err != nil {
return nil, errz.Wrapf(err, "unable to open XLSX file: %s", d.src.Location)
}
hasHeader, _, err := options.HasHeader(d.src.Options)
if err != nil {
return nil, err
}
for _, sheet := range xlFile.Sheets {
tbl := &source.TableMetadata{Name: sheet.Name, RowCount: int64(len(sheet.Rows))}
if hasHeader && tbl.RowCount > 0 {
tbl.RowCount--
}
colNames := getColNames(sheet, hasHeader)
// TODO: Should move over to using kind.Detector
colTypes := getCellColumnTypes(sheet, hasHeader)
for i, colType := range colTypes {
col := &source.ColMetadata{}
col.BaseType = cellTypeToString(colType)
col.ColumnType = col.BaseType
col.Position = int64(i)
col.Name = colNames[i]
tbl.Columns = append(tbl.Columns, col)
}
meta.Tables = append(meta.Tables, tbl)
}
return meta, nil
}
// TableMetadata implements driver.Database.
func (d *database) TableMetadata(_ context.Context, tblName string) (*source.TableMetadata, error) {
b, err := d.files.ReadAll(d.src)
if err != nil {
return nil, errz.Err(err)
}
xlFile, err := xlsx.OpenBinary(b)
if err != nil {
return nil, errz.Wrapf(err, "unable to open XLSX file: %s", d.src.Location)
}
hasHeader, _, err := options.HasHeader(d.src.Options)
if err != nil {
return nil, err
}
for _, sheet := range xlFile.Sheets {
if sheet.Name != tblName {
continue
}
tbl := &source.TableMetadata{Name: sheet.Name, RowCount: int64(len(sheet.Rows))}
if hasHeader && tbl.RowCount > 0 {
tbl.RowCount--
}
colNames := getColNames(sheet, hasHeader)
// TODO: Should move over to using kind.Detector
colTypes := getCellColumnTypes(sheet, hasHeader)
for i, colType := range colTypes {
col := &source.ColMetadata{}
col.BaseType = cellTypeToString(colType)
col.ColumnType = col.BaseType
col.Position = int64(i)
col.Name = colNames[i]
tbl.Columns = append(tbl.Columns, col)
}
return tbl, nil
}
return nil, errz.Errorf("table {%s} not found", tblName)
}
// Close implements driver.Database.
func (d *database) Close() error {
d.log.Debug(lgm.CloseDB, lga.Src, d.src)
// No need to explicitly invoke c.impl.Close because
// that's already added to c.clnup
return d.clnup.Run()
}