From f73c926fb2a993f928cbdf0f184500262b093cd0 Mon Sep 17 00:00:00 2001 From: Michal Sokolowski <0michalsokolowski0@gmail.com> Date: Fri, 30 Aug 2024 15:02:31 +0200 Subject: [PATCH] feat: Add limit and search flags to stack list command (#252) --- internal/cmd/stack/flags.go | 10 ++++ internal/cmd/stack/list.go | 104 +++++++++++++++++++++++++++++++----- internal/cmd/stack/stack.go | 2 + internal/helpers.go | 5 ++ 4 files changed, 109 insertions(+), 12 deletions(-) create mode 100644 internal/helpers.go diff --git a/internal/cmd/stack/flags.go b/internal/cmd/stack/flags.go index 55cc426..140fdca 100644 --- a/internal/cmd/stack/flags.go +++ b/internal/cmd/stack/flags.go @@ -157,3 +157,13 @@ var flagInteractive = &cli.BoolFlag{ Aliases: []string{"i"}, Usage: "[Optional] Whether to run the command in interactive mode", } + +var flagLimit = &cli.UintFlag{ + Name: "limit", + Usage: "[Optional] Limit the number of items to return", +} + +var flagSearch = &cli.StringFlag{ + Name: "search", + Usage: "[Optional] Performs a full-text search.", +} diff --git a/internal/cmd/stack/list.go b/internal/cmd/stack/list.go index 32a3f19..9386038 100644 --- a/internal/cmd/stack/list.go +++ b/internal/cmd/stack/list.go @@ -3,10 +3,13 @@ package stack import ( "context" "fmt" + "math" + "slices" "strings" "github.com/shurcooL/graphql" "github.com/spacelift-io/spacectl/client/structs" + "github.com/spacelift-io/spacectl/internal" "github.com/spacelift-io/spacectl/internal/cmd" "github.com/urfave/cli/v2" ) @@ -18,20 +21,57 @@ func listStacks() cli.ActionFunc { return err } + var limit *uint + if cliCtx.IsSet(flagLimit.Name) { + if cliCtx.Uint(flagLimit.Name) == 0 { + return fmt.Errorf("limit must be greater than 0") + } + + if cliCtx.Uint(flagLimit.Name) >= math.MaxInt32 { + return fmt.Errorf("limit must be less than %d", math.MaxInt32) + } + + limit = internal.Ptr(cliCtx.Uint(flagLimit.Name)) + } + + var search *string + if cliCtx.IsSet(flagSearch.Name) { + if cliCtx.String(flagSearch.Name) == "" { + return fmt.Errorf("search must be non-empty") + } + + search = internal.Ptr(cliCtx.String(flagSearch.Name)) + } + switch outputFormat { case cmd.OutputFormatTable: - return listStacksTable(cliCtx) + return listStacksTable(cliCtx, search, limit) case cmd.OutputFormatJSON: - return listStacksJSON(cliCtx) + return listStacksJSON(cliCtx, search, limit) } return fmt.Errorf("unknown output format: %v", outputFormat) } } -func listStacksJSON(ctx *cli.Context) error { +func listStacksJSON( + ctx *cli.Context, + search *string, + limit *uint, +) error { + var first *graphql.Int + if limit != nil { + first = graphql.NewInt(graphql.Int(*limit)) //nolint: gosec + } + + var fullTextSearch *graphql.String + if search != nil { + fullTextSearch = graphql.NewString(graphql.String(*search)) + } + stacks, err := searchAllStacks(ctx.Context, structs.SearchInput{ - First: graphql.NewInt(50), + First: first, + FullTextSearch: fullTextSearch, }) if err != nil { return err @@ -40,14 +80,31 @@ func listStacksJSON(ctx *cli.Context) error { return cmd.OutputJSON(stacks) } -func listStacksTable(ctx *cli.Context) error { - stacks, err := searchAllStacks(ctx.Context, structs.SearchInput{ - First: graphql.NewInt(50), +func listStacksTable( + ctx *cli.Context, + search *string, + limit *uint, +) error { + var first *graphql.Int + if limit != nil { + first = graphql.NewInt(graphql.Int(*limit)) //nolint: gosec + } + + var fullTextSearch *graphql.String + if search != nil { + fullTextSearch = graphql.NewString(graphql.String(*search)) + } + + input := structs.SearchInput{ + First: first, + FullTextSearch: fullTextSearch, OrderBy: &structs.QueryOrder{ Field: "starred", Direction: "DESC", }, - }) + } + + stacks, err := searchAllStacks(ctx.Context, input) if err != nil { return err } @@ -78,19 +135,42 @@ func listStacksTable(ctx *cli.Context) error { return cmd.OutputTable(tableData, true) } +// searchStacks returns a list of stacks based on the provided search input. +// input.First limits the total number of returned stacks, if not provided all stacks are returned. func searchAllStacks(ctx context.Context, input structs.SearchInput) ([]stack, error) { - out := []stack{} + const maxPageSize = 50 + var limit int + if input.First != nil { + limit = int(*input.First) + } + fetchAll := limit == 0 + + out := []stack{} + pageInput := structs.SearchInput{ + First: graphql.NewInt(maxPageSize), + FullTextSearch: input.FullTextSearch, + } for { - result, err := searchStacks(ctx, input) + if !fetchAll { + // Fetch exactly the number of items requested + pageInput.First = graphql.NewInt( + //nolint: gosec + graphql.Int( + slices.Min([]int{maxPageSize, limit - len(out)}), + ), + ) + } + + result, err := searchStacks(ctx, pageInput) if err != nil { return nil, err } out = append(out, result.Stacks...) - if result.PageInfo.HasNextPage { - input.After = graphql.NewString(graphql.String(result.PageInfo.EndCursor)) + if result.PageInfo.HasNextPage && (fetchAll || limit > len(out)) { + pageInput.After = graphql.NewString(graphql.String(result.PageInfo.EndCursor)) } else { break } diff --git a/internal/cmd/stack/stack.go b/internal/cmd/stack/stack.go index 0575283..adf2c3d 100644 --- a/internal/cmd/stack/stack.go +++ b/internal/cmd/stack/stack.go @@ -140,6 +140,8 @@ func Command() *cli.Command { flagShowLabels, cmd.FlagOutputFormat, cmd.FlagNoColor, + flagLimit, + flagSearch, }, Action: listStacks(), Before: cmd.PerformAllBefore(cmd.HandleNoColor, authenticated.Ensure), diff --git a/internal/helpers.go b/internal/helpers.go new file mode 100644 index 0000000..729f5f3 --- /dev/null +++ b/internal/helpers.go @@ -0,0 +1,5 @@ +package internal + +func Ptr[T any](v T) *T { + return &v +}