2023-11-19 03:05:48 +03:00
|
|
|
package cli_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
2023-11-20 04:06:36 +03:00
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
2024-01-25 07:01:24 +03:00
|
|
|
"github.com/neilotoole/sq/cli"
|
2023-11-19 03:05:48 +03:00
|
|
|
"github.com/neilotoole/sq/cli/cobraz"
|
|
|
|
"github.com/neilotoole/sq/cli/flag"
|
2023-11-20 04:06:36 +03:00
|
|
|
"github.com/neilotoole/sq/cli/testrun"
|
2023-11-19 03:05:48 +03:00
|
|
|
"github.com/neilotoole/sq/libsq/core/lg"
|
2024-01-15 04:45:34 +03:00
|
|
|
"github.com/neilotoole/sq/libsq/core/lg/lgt"
|
2024-01-25 07:01:24 +03:00
|
|
|
"github.com/neilotoole/sq/libsq/core/options"
|
2023-11-19 03:05:48 +03:00
|
|
|
"github.com/neilotoole/sq/testh"
|
2023-11-20 04:06:36 +03:00
|
|
|
"github.com/neilotoole/sq/testh/sakila"
|
2024-01-15 04:45:34 +03:00
|
|
|
"github.com/neilotoole/sq/testh/tu"
|
2023-11-19 03:05:48 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// testComplete is a helper for testing cobra completion.
|
|
|
|
func testComplete(t testing.TB, from *testrun.TestRun, args ...string) completion {
|
2024-01-25 07:01:24 +03:00
|
|
|
var ctx context.Context
|
|
|
|
if from == nil {
|
|
|
|
ctx = lg.NewContext(context.Background(), lgt.New(t))
|
|
|
|
} else {
|
|
|
|
ctx = from.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable completion logging.
|
|
|
|
ctx = options.NewContext(ctx, options.Options{cli.OptShellCompletionLog.Key(): true})
|
2023-11-19 03:05:48 +03:00
|
|
|
|
|
|
|
tr := testrun.New(ctx, t, from)
|
2024-01-25 07:01:24 +03:00
|
|
|
args = append([]string{cobra.ShellCompRequestCmd}, args...)
|
2023-11-19 03:05:48 +03:00
|
|
|
|
|
|
|
err := tr.Exec(args...)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
c := parseCompletion(tr)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
// parseCompletion parses the output of cobra "__complete".
|
|
|
|
// Example output:
|
|
|
|
//
|
|
|
|
// @active
|
|
|
|
// @sakila
|
|
|
|
// :4
|
|
|
|
// Completion ended with directive: ShellCompDirectiveNoFileComp
|
|
|
|
//
|
|
|
|
// The tr.T test will fail on any error.
|
|
|
|
func parseCompletion(tr *testrun.TestRun) completion {
|
|
|
|
c := completion{
|
|
|
|
stdout: tr.Out.String(),
|
|
|
|
stderr: tr.ErrOut.String(),
|
|
|
|
}
|
|
|
|
|
|
|
|
lines := strings.Split(strings.TrimSpace(c.stdout), "\n")
|
|
|
|
require.True(tr.T, len(lines) >= 1)
|
|
|
|
c.values = lines[:len(lines)-1]
|
|
|
|
|
|
|
|
result, err := strconv.Atoi(lines[len(lines)-1][1:])
|
|
|
|
require.NoError(tr.T, err)
|
|
|
|
c.result = cobra.ShellCompDirective(result)
|
|
|
|
|
|
|
|
c.directives = cobraz.ParseDirectivesLine(c.stderr)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
// completion models the result returned from the cobra "__complete" command.
|
|
|
|
type completion struct {
|
|
|
|
stdout string
|
|
|
|
stderr string
|
|
|
|
values []string
|
|
|
|
result cobra.ShellCompDirective
|
|
|
|
directives []cobra.ShellCompDirective
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestCompleteFlagActiveSchema_query_cmds tests flag.ActiveSchema
|
|
|
|
// behavior for the query commands (slq, sql).
|
|
|
|
//
|
|
|
|
// See also: TestCompleteFlagActiveSchema_inspect.
|
2024-01-25 07:01:24 +03:00
|
|
|
func TestCompleteFlagActiveSchema_query_cmds(t *testing.T) { //nolint:tparallel
|
2023-11-19 03:05:48 +03:00
|
|
|
const wantDirective = cobra.ShellCompDirectiveNoFileComp | cobra.ShellCompDirectiveKeepOrder
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
handles []string
|
|
|
|
arg string
|
|
|
|
withFlagActiveSrc string
|
|
|
|
wantContains []string
|
|
|
|
wantDirective cobra.ShellCompDirective
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "saki",
|
|
|
|
wantContains: []string{"sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "",
|
|
|
|
wantContains: []string{"public", "sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "sakila",
|
|
|
|
wantContains: []string{"sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "sakila.pub",
|
|
|
|
wantContains: []string{"sakila.public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "pub",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "public",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.My, sakila.Pg},
|
|
|
|
withFlagActiveSrc: sakila.Pg,
|
|
|
|
arg: "publ",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg, sakila.My},
|
|
|
|
withFlagActiveSrc: sakila.My,
|
|
|
|
arg: "",
|
|
|
|
wantContains: []string{"mysql", "sys", "information_schema", "sakila"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.My, sakila.Pg},
|
|
|
|
withFlagActiveSrc: sakila.MS,
|
|
|
|
arg: "publ",
|
|
|
|
// Should error because sakila.MS isn't a loaded source (via "handles").
|
|
|
|
wantDirective: cobra.ShellCompDirectiveError,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cmdName := range []string{"slq", "sql"} {
|
|
|
|
cmdName := cmdName
|
|
|
|
t.Run(cmdName, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
for i, tc := range testCases {
|
|
|
|
tc := tc
|
2024-01-15 04:45:34 +03:00
|
|
|
t.Run(tu.Name(i, tc.handles, tc.withFlagActiveSrc, tc.arg), func(t *testing.T) {
|
2023-11-19 03:05:48 +03:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
th := testh.New(t)
|
|
|
|
tr := testrun.New(th.Context, t, nil)
|
|
|
|
for _, handle := range tc.handles {
|
|
|
|
tr.Add(*th.Source(handle))
|
|
|
|
}
|
|
|
|
|
|
|
|
args := []string{cmdName}
|
|
|
|
if tc.withFlagActiveSrc != "" {
|
|
|
|
args = append(args, "--"+flag.ActiveSrc, tc.withFlagActiveSrc)
|
|
|
|
}
|
|
|
|
args = append(args, "--"+flag.ActiveSchema, tc.arg)
|
|
|
|
|
|
|
|
got := testComplete(t, tr, args...)
|
|
|
|
assert.Equal(t, tc.wantDirective, got.result,
|
|
|
|
"wanted: %v\ngot : %v",
|
|
|
|
cobraz.MarshalDirective(tc.wantDirective),
|
|
|
|
cobraz.MarshalDirective(got.result))
|
|
|
|
|
|
|
|
if tc.wantDirective == cobra.ShellCompDirectiveError {
|
|
|
|
require.Empty(t, got.values)
|
|
|
|
} else {
|
|
|
|
for j := range tc.wantContains {
|
|
|
|
assert.Contains(t, got.values, tc.wantContains[j])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestCompleteFlagActiveSchema_inspect tests flag.ActiveSchema
|
|
|
|
// behavior for the inspect command.
|
|
|
|
//
|
|
|
|
// See also: TestCompleteFlagActiveSchema_query_cmds.
|
|
|
|
func TestCompleteFlagActiveSchema_inspect(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
const wantDirective = cobra.ShellCompDirectiveNoFileComp | cobra.ShellCompDirectiveKeepOrder
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
handles []string
|
|
|
|
arg string
|
|
|
|
withArgActiveSrc string
|
|
|
|
wantContains []string
|
|
|
|
wantDirective cobra.ShellCompDirective
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "saki",
|
|
|
|
wantContains: []string{"sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "",
|
|
|
|
wantContains: []string{"public", "sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "sakila",
|
|
|
|
wantContains: []string{"sakila."},
|
|
|
|
wantDirective: wantDirective | cobra.ShellCompDirectiveNoSpace,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "sakila.pub",
|
|
|
|
wantContains: []string{"sakila.public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "pub",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg},
|
|
|
|
arg: "public",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.My, sakila.Pg},
|
|
|
|
withArgActiveSrc: sakila.Pg,
|
|
|
|
arg: "publ",
|
|
|
|
wantContains: []string{"public"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.Pg, sakila.My},
|
|
|
|
withArgActiveSrc: sakila.My,
|
|
|
|
arg: "",
|
|
|
|
wantContains: []string{"mysql", "sys", "information_schema", "sakila"},
|
|
|
|
wantDirective: wantDirective,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
handles: []string{sakila.My, sakila.Pg},
|
|
|
|
withArgActiveSrc: sakila.MS,
|
|
|
|
arg: "publ",
|
|
|
|
// Should error because sakila.MS isn't a loaded source (via "handles").
|
|
|
|
wantDirective: cobra.ShellCompDirectiveError,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range testCases {
|
|
|
|
tc := tc
|
2024-01-15 04:45:34 +03:00
|
|
|
t.Run(tu.Name(i, tc.handles, tc.withArgActiveSrc, tc.arg), func(t *testing.T) {
|
2023-11-19 03:05:48 +03:00
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
th := testh.New(t)
|
|
|
|
tr := testrun.New(th.Context, t, nil)
|
|
|
|
for _, handle := range tc.handles {
|
|
|
|
tr.Add(*th.Source(handle))
|
|
|
|
}
|
|
|
|
|
|
|
|
args := []string{"inspect"}
|
|
|
|
if tc.withArgActiveSrc != "" {
|
|
|
|
args = append(args, tc.withArgActiveSrc)
|
|
|
|
}
|
|
|
|
args = append(args, "--"+flag.ActiveSchema, tc.arg)
|
|
|
|
|
|
|
|
got := testComplete(t, tr, args...)
|
|
|
|
assert.Equal(t, tc.wantDirective, got.result,
|
|
|
|
"wanted: %v\ngot : %v",
|
|
|
|
cobraz.MarshalDirective(tc.wantDirective),
|
|
|
|
cobraz.MarshalDirective(got.result))
|
|
|
|
|
|
|
|
if tc.wantDirective == cobra.ShellCompDirectiveError {
|
|
|
|
require.Empty(t, got.values)
|
|
|
|
} else {
|
|
|
|
for j := range tc.wantContains {
|
|
|
|
assert.Contains(t, got.values, tc.wantContains[j])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|