sq/cli/cmd_sql_test.go

230 lines
6.1 KiB
Go
Raw Normal View History

2020-08-06 20:58:47 +03:00
package cli_test
import (
2023-04-22 16:37:07 +03:00
"context"
2020-08-06 20:58:47 +03:00
"fmt"
"os"
"path/filepath"
2020-08-06 20:58:47 +03:00
"strings"
"testing"
"github.com/stretchr/testify/require"
"github.com/neilotoole/sq/cli/flag"
"github.com/neilotoole/sq/cli/testrun"
2020-08-06 20:58:47 +03:00
"github.com/neilotoole/sq/drivers/userdriver"
"github.com/neilotoole/sq/libsq/core/tablefq"
2020-08-06 20:58:47 +03:00
"github.com/neilotoole/sq/libsq/source"
"github.com/neilotoole/sq/testh"
"github.com/neilotoole/sq/testh/proj"
"github.com/neilotoole/sq/testh/sakila"
"github.com/neilotoole/sq/testh/testsrc"
"github.com/neilotoole/sq/testh/tu"
2020-08-06 20:58:47 +03:00
)
// TestCmdSQL_Insert tests "sq sql QUERY --insert=dest.tbl".
2020-08-06 20:58:47 +03:00
func TestCmdSQL_Insert(t *testing.T) {
for _, origin := range sakila.SQLLatest() {
2020-08-06 20:58:47 +03:00
origin := origin
t.Run("origin_"+origin, func(t *testing.T) {
tu.SkipShort(t, origin == sakila.XLSX)
2020-08-06 20:58:47 +03:00
for _, dest := range sakila.SQLLatest() {
2020-08-06 20:58:47 +03:00
dest := dest
t.Run("dest_"+dest, func(t *testing.T) {
t.Parallel()
th := testh.New(t)
originSrc, destSrc := th.Source(origin), th.Source(dest)
originTbl := sakila.TblActor
if th.IsMonotable(originSrc) {
originTbl = source.MonotableName
}
// To avoid dirtying the destination table, we make a copy
// of it (without data).
destTbl := th.CopyTable(true, destSrc, tablefq.From(sakila.TblActor), tablefq.T{}, false)
2020-08-06 20:58:47 +03:00
tr := testrun.New(th.Context, t, nil).Add(*originSrc)
2020-08-06 20:58:47 +03:00
if destSrc.Handle != originSrc.Handle {
tr.Add(*destSrc)
2020-08-06 20:58:47 +03:00
}
insertTo := fmt.Sprintf("%s.%s", destSrc.Handle, destTbl)
query := fmt.Sprintf("SELECT %s FROM %s", strings.Join(sakila.TblActorCols(), ", "), originTbl)
2020-08-06 20:58:47 +03:00
err := tr.Exec("sql", "--insert="+insertTo, query)
2020-08-06 20:58:47 +03:00
require.NoError(t, err)
sink, err := th.QuerySQL(destSrc, nil, "select * from "+destTbl)
2020-08-06 20:58:47 +03:00
require.NoError(t, err)
require.Equal(t, sakila.TblActorCount, len(sink.Recs))
})
}
})
}
}
func TestCmdSQL_SelectFromUserDriver(t *testing.T) {
t.Parallel()
2020-08-06 20:58:47 +03:00
testCases := map[string][]struct {
tblName string
wantRows int
wantCols int
}{
testsrc.PplUD: {
{tblName: "person", wantRows: 3, wantCols: 7},
{tblName: "skill", wantRows: 6, wantCols: 3},
},
testsrc.RSSNYTLocalUD: {
{tblName: "category", wantRows: 251, wantCols: 4},
{tblName: "channel", wantRows: 1, wantCols: 7},
{tblName: "item", wantRows: 45, wantCols: 9},
},
}
for handle, wantTbls := range testCases {
for _, wantTbl := range wantTbls {
handle, wantTbl := handle, wantTbl
t.Run(handle+"__"+wantTbl.tblName, func(t *testing.T) {
t.Parallel()
th := testh.New(t)
src := th.Source(handle)
tr := testrun.New(th.Context, t, nil).Add(*src)
2020-08-06 20:58:47 +03:00
udDefs := testh.DriverDefsFrom(t, testsrc.PathDriverDefPpl, testsrc.PathDriverDefRSS)
require.Len(t, udDefs, 2)
2020-08-06 20:58:47 +03:00
for _, udDef := range udDefs {
require.Empty(t, userdriver.ValidateDriverDef(udDef))
}
tr.Run.Config.Ext.UserDrivers = append(tr.Run.Config.Ext.UserDrivers, udDefs...)
2020-08-06 20:58:47 +03:00
err := tr.Exec("sql", "--csv", "--header=false", "SELECT * FROM "+wantTbl.tblName)
2020-08-06 20:58:47 +03:00
require.NoError(t, err)
recs := tr.BindCSV()
require.Equal(t, wantTbl.wantRows, len(recs),
"expected %d rows in tbl {%s} but got %d", wantTbl.wantRows,
wantTbl, len(recs))
require.Equal(t, wantTbl.wantCols, len(recs[0]),
"expected %d cols in tbl {%s} but got %d",
wantTbl.wantCols, wantTbl, len(recs[0]))
2020-08-06 20:58:47 +03:00
})
}
}
}
// TestCmdSQL_StdinQuery verifies that cmd sql can read from stdin.
func TestCmdSQL_StdinQuery(t *testing.T) {
t.Parallel()
testCases := []struct {
fpath string
tbl string
flags map[string]string
2020-08-06 20:58:47 +03:00
wantCount int
wantErr bool
}{
{
fpath: proj.Abs(sakila.PathCSVActorNoHeader),
flags: map[string]string{flag.IngestHeader: "false"},
tbl: source.MonotableName,
wantCount: sakila.TblActorCount,
},
{
fpath: proj.Abs(sakila.PathCSVActor),
flags: map[string]string{flag.IngestHeader: "true"},
tbl: source.MonotableName,
wantCount: sakila.TblActorCount,
},
{
fpath: proj.Abs(sakila.PathXLSXActorHeader),
flags: map[string]string{flag.IngestHeader: "true"},
tbl: sakila.TblActor,
wantCount: sakila.TblActorCount,
},
{
fpath: proj.Abs(sakila.PathXLSXSubset),
flags: map[string]string{flag.IngestHeader: "true"},
tbl: sakila.TblActor,
wantCount: sakila.TblActorCount,
},
{
fpath: proj.Abs("README.md"),
wantErr: true,
},
2020-08-06 20:58:47 +03:00
}
for i, tc := range testCases {
2020-08-06 20:58:47 +03:00
tc := tc
name := tu.Name(i, filepath.Base(filepath.Dir(tc.fpath)), filepath.Base(tc.fpath))
t.Run(name, func(t *testing.T) {
2020-08-06 20:58:47 +03:00
t.Parallel()
f, err := os.Open(tc.fpath)
require.NoError(t, err)
tr := testrun.New(context.Background(), t, nil).Hush()
tr.Run.Stdin = f
2020-08-06 20:58:47 +03:00
args := []string{"sql", "--header=false"} // Don't print the header in output
for k, v := range tc.flags {
args = append(args, fmt.Sprintf("--%s=%s", k, v))
}
args = append(args, "SELECT * FROM "+tc.tbl)
err = tr.Exec(args...)
2020-08-06 20:58:47 +03:00
if tc.wantErr {
require.Error(t, err)
return
}
require.NoError(t, err)
results := tr.BindCSV()
2020-08-06 20:58:47 +03:00
require.Equal(t, tc.wantCount, len(results))
})
}
}
func TestFlagActiveSource_sql(t *testing.T) {
t.Parallel()
ctx := context.Background()
tr := testrun.New(ctx, t, nil)
// @sqlite will be the active source
require.NoError(t, tr.Exec("add", proj.Abs(sakila.PathSL3), "--handle", "@sqlite"))
tr = testrun.New(ctx, t, tr)
require.NoError(t, tr.Exec("add", proj.Abs(sakila.PathCSVActor), "--handle", "@csv"))
tr = testrun.New(ctx, t, tr)
require.NoError(t, tr.Exec(
"sql",
"--csv",
"--no-header",
`select * from actor`,
))
require.Len(t, tr.BindCSV(), sakila.TblActorCount)
// Now, use flag.ActiveSrc to switch the source.
tr = testrun.New(ctx, t, tr)
require.NoError(t, tr.Exec(
"sql",
"--csv",
"--no-header",
"--src", "@csv",
"select * from data",
))
require.Len(t, tr.BindCSV(), sakila.TblActorCount)
// Double check that we didn't change the persisted active source
tr = testrun.New(ctx, t, tr)
require.NoError(t, tr.Exec("src", "--json"))
require.Equal(t, "@sqlite", tr.BindMap()["handle"])
}