Skip to content

Commit

Permalink
Go: Add HScan command (valkey-io#2917)
Browse files Browse the repository at this point in the history
* Go: Add HScan command

Signed-off-by: Prateek Kumar <[email protected]>
  • Loading branch information
prateek-kumar-improving authored Jan 6, 2025
1 parent 2bd6893 commit 8b904d4
Show file tree
Hide file tree
Showing 10 changed files with 379 additions and 52 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
#### Changes

* Go: Add `HScan` command ([#2917](https://github.com/valkey-io/valkey-glide/pull/2917))
* Java, Node, Python: Add transaction commands for JSON module ([#2862](https://github.com/valkey-io/valkey-glide/pull/2862))
* Go: Add HINCRBY command ([#2847](https://github.com/valkey-io/valkey-glide/pull/2847))
* Go: Add HINCRBYFLOAT command ([#2846](https://github.com/valkey-io/valkey-glide/pull/2846))
Expand Down
29 changes: 27 additions & 2 deletions go/api/base_client.go
Original file line number Diff line number Diff line change
Expand Up @@ -484,6 +484,31 @@ func (client *baseClient) HIncrByFloat(key string, field string, increment float
return handleDoubleResponse(result)
}

func (client *baseClient) HScan(key string, cursor string) (Result[string], []Result[string], error) {
result, err := client.executeCommand(C.HScan, []string{key, cursor})
if err != nil {
return CreateNilStringResult(), nil, err
}
return handleScanResponse(result)
}

func (client *baseClient) HScanWithOptions(
key string,
cursor string,
options *options.HashScanOptions,
) (Result[string], []Result[string], error) {
optionArgs, err := options.ToArgs()
if err != nil {
return CreateNilStringResult(), nil, err
}

result, err := client.executeCommand(C.HScan, append([]string{key, cursor}, optionArgs...))
if err != nil {
return CreateNilStringResult(), nil, err
}
return handleScanResponse(result)
}

func (client *baseClient) LPush(key string, elements []string) (Result[int64], error) {
result, err := client.executeCommand(C.LPush, append([]string{key}, elements...))
if err != nil {
Expand Down Expand Up @@ -721,9 +746,9 @@ func (client *baseClient) SScan(key string, cursor string) (Result[string], []Re
func (client *baseClient) SScanWithOptions(
key string,
cursor string,
options *BaseScanOptions,
options *options.BaseScanOptions,
) (Result[string], []Result[string], error) {
optionArgs, err := options.toArgs()
optionArgs, err := options.ToArgs()
if err != nil {
return CreateNilStringResult(), nil, err
}
Expand Down
43 changes: 0 additions & 43 deletions go/api/command_options.go
Original file line number Diff line number Diff line change
Expand Up @@ -278,46 +278,3 @@ func (listDirection ListDirection) toString() (string, error) {
return "", &RequestError{"Invalid list direction"}
}
}

// This base option struct represents the common set of optional arguments for the SCAN family of commands.
// Concrete implementations of this class are tied to specific SCAN commands (`SCAN`, `SSCAN`).
type BaseScanOptions struct {
match string
count int64
}

func NewBaseScanOptionsBuilder() *BaseScanOptions {
return &BaseScanOptions{}
}

// The match filter is applied to the result of the command and will only include
// strings that match the pattern specified. If the sorted set is large enough for scan commands to return
// only a subset of the sorted set then there could be a case where the result is empty although there are
// items that match the pattern specified. This is due to the default `COUNT` being `10` which indicates
// that it will only fetch and match `10` items from the list.
func (scanOptions *BaseScanOptions) SetMatch(m string) *BaseScanOptions {
scanOptions.match = m
return scanOptions
}

// `COUNT` is a just a hint for the command for how many elements to fetch from the
// sorted set. `COUNT` could be ignored until the sorted set is large enough for the `SCAN` commands to
// represent the results as compact single-allocation packed encoding.
func (scanOptions *BaseScanOptions) SetCount(c int64) *BaseScanOptions {
scanOptions.count = c
return scanOptions
}

func (opts *BaseScanOptions) toArgs() ([]string, error) {
args := []string{}
var err error
if opts.match != "" {
args = append(args, MatchKeyword, opts.match)
}

if opts.count != 0 {
args = append(args, CountKeyword, strconv.FormatInt(opts.count, 10))
}

return args, err
}
56 changes: 56 additions & 0 deletions go/api/hash_commands.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@

package api

import "github.com/valkey-io/valkey-glide/go/glide/api/options"

// Supports commands and transactions for the "Hash" group of commands for standalone and cluster clients.
//
// See [valkey.io] for details.
Expand Down Expand Up @@ -292,4 +294,58 @@ type HashCommands interface {
//
// [valkey.io]: https://valkey.io/commands/hincrbyfloat/
HIncrByFloat(key string, field string, increment float64) (Result[float64], error)

// Iterates fields of Hash types and their associated values. This definition of HSCAN command does not include the
// optional arguments of the command.
//
// See [valkey.io] for details.
//
// Parameters:
// key - The key of the hash.
// cursor - The cursor that points to the next iteration of results. A value of "0" indicates the start of the search.
//
// Return value:
// An array of the cursor and the subset of the hash held by `key`. The first element is always the `cursor`
// for the next iteration of results. The `cursor` will be `"0"` on the last iteration of the subset.
// The second element is always an array of the subset of the set held in `key`. The array in the
// second element is always a flattened series of String pairs, where the key is at even indices
// and the value is at odd indices.
//
// Example:
// // Assume key contains a hash {{"a": "1"}, {"b", "2"}}
// resCursor, resCollection, err = client.HScan(key, initialCursor)
// // resCursor = {0 false}
// // resCollection = [{a false} {1 false} {b false} {2 false}]
//
// [valkey.io]: https://valkey.io/commands/hscan/
HScan(key string, cursor string) (Result[string], []Result[string], error)

// Iterates fields of Hash types and their associated values. This definition of HSCAN includes optional arguments of the
// command.
//
// See [valkey.io] for details.
//
// Parameters:
// key - The key of the hash.
// cursor - The cursor that points to the next iteration of results. A value of "0" indicates the start of the search.
// options - The [api.HashScanOptions].
//
// Return value:
// An array of the cursor and the subset of the hash held by `key`. The first element is always the `cursor`
// for the next iteration of results. The `cursor` will be `"0"` on the last iteration of the subset.
// The second element is always an array of the subset of the set held in `key`. The array in the
// second element is always a flattened series of String pairs, where the key is at even indices
// and the value is at odd indices.
//
// Example:
// // Assume key contains a hash {{"a": "1"}, {"b", "2"}}
// opts := options.NewHashScanOptionsBuilder().SetMatch("a")
// resCursor, resCollection, err = client.HScan(key, initialCursor, opts)
// // resCursor = {0 false}
// // resCollection = [{a false} {1 false}]
// // The resCollection only contains the hash map entry that matches with the match option provided with the command
// // input.
//
// [valkey.io]: https://valkey.io/commands/hscan/
HScanWithOptions(key string, cursor string, options *options.HashScanOptions) (Result[string], []Result[string], error)
}
54 changes: 54 additions & 0 deletions go/api/options/base_scan_options.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
// Copyright Valkey GLIDE Project Contributors - SPDX Identifier: Apache-2.0

package options

import (
"strconv"
)

// This base option struct represents the common set of optional arguments for the SCAN family of commands.
// Concrete implementations of this class are tied to specific SCAN commands (`SCAN`, `SSCAN`, `HSCAN`).
type BaseScanOptions struct {
match string
count int64
}

func NewBaseScanOptionsBuilder() *BaseScanOptions {
return &BaseScanOptions{}
}

/*
The match filter is applied to the result of the command and will only include
strings that match the pattern specified. If the sorted set is large enough for scan commands to return
only a subset of the sorted set then there could be a case where the result is empty although there are
items that match the pattern specified. This is due to the default `COUNT` being `10` which indicates
that it will only fetch and match `10` items from the list.
*/
func (scanOptions *BaseScanOptions) SetMatch(m string) *BaseScanOptions {
scanOptions.match = m
return scanOptions
}

/*
`COUNT` is a just a hint for the command for how many elements to fetch from the
sorted set. `COUNT` could be ignored until the sorted set is large enough for the `SCAN` commands to
represent the results as compact single-allocation packed encoding.
*/
func (scanOptions *BaseScanOptions) SetCount(c int64) *BaseScanOptions {
scanOptions.count = c
return scanOptions
}

func (opts *BaseScanOptions) ToArgs() ([]string, error) {
args := []string{}
var err error
if opts.match != "" {
args = append(args, MatchKeyword, opts.match)
}

if opts.count != 0 {
args = append(args, CountKeyword, strconv.FormatInt(opts.count, 10))
}

return args, err
}
9 changes: 9 additions & 0 deletions go/api/options/constants.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
// Copyright Valkey GLIDE Project Contributors - SPDX Identifier: Apache-2.0

package options

const (
CountKeyword string = "COUNT" // Valkey API keyword used to extract specific number of matching indices from a list.
MatchKeyword string = "MATCH" // Valkey API keyword used to indicate the match filter.
NoValue string = "NOVALUE" // Valkey API keyword for the no value option for hcsan command.
)
43 changes: 43 additions & 0 deletions go/api/options/hscan_options.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
// Copyright Valkey GLIDE Project Contributors - SPDX Identifier: Apache-2.0

package options

// This struct represents the optional arguments for the HSCAN command.
type HashScanOptions struct {
BaseScanOptions
noValue bool
}

func NewHashScanOptionsBuilder() *HashScanOptions {
return &HashScanOptions{}
}

/*
If this value is set to true, the HSCAN command will be called with NOVALUES option.
In the NOVALUES option, values are not included in the response.
*/
func (hashScanOptions *HashScanOptions) SetNoValue(noValue bool) *HashScanOptions {
hashScanOptions.noValue = noValue
return hashScanOptions
}

func (hashScanOptions *HashScanOptions) SetMatch(match string) *HashScanOptions {
hashScanOptions.BaseScanOptions.SetMatch(match)
return hashScanOptions
}

func (hashScanOptions *HashScanOptions) SetCount(count int64) *HashScanOptions {
hashScanOptions.BaseScanOptions.SetCount(count)
return hashScanOptions
}

func (options *HashScanOptions) ToArgs() ([]string, error) {
args := []string{}
baseArgs, err := options.BaseScanOptions.ToArgs()
args = append(args, baseArgs...)

if options.noValue {
args = append(args, NoValue)
}
return args, err
}
8 changes: 5 additions & 3 deletions go/api/set_commands.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@

package api

import "github.com/valkey-io/valkey-glide/go/glide/api/options"

// Supports commands and transactions for the "Set" group of commands for standalone and cluster clients.
//
// See [valkey.io] for details.
Expand Down Expand Up @@ -429,7 +431,7 @@ type SetCommands interface {
// cursor - The cursor that points to the next iteration of results.
// A value of `"0"` indicates the start of the search.
// For Valkey 8.0 and above, negative cursors are treated like the initial cursor("0").
// options - [BaseScanOptions]
// options - [options.BaseScanOptions]
//
// Return value:
// An array of the cursor and the subset of the set held by `key`. The first element is always the `cursor` and
Expand All @@ -440,7 +442,7 @@ type SetCommands interface {
// // assume "key" contains a set
// resCursor resCol, err := client.sscan("key", "0", opts)
// for resCursor != "0" {
// opts := api.NewBaseScanOptionsBuilder().SetMatch("*")
// opts := options.NewBaseScanOptionsBuilder().SetMatch("*")
// resCursor, resCol, err = client.sscan("key", "0", opts)
// fmt.Println("Cursor: ", resCursor.Value())
// fmt.Println("Members: ", resCol.Value())
Expand All @@ -454,7 +456,7 @@ type SetCommands interface {
// // Members: ['47', '122', '1', '53', '10', '14', '80']
//
// [valkey.io]: https://valkey.io/commands/sscan/
SScanWithOptions(key string, cursor string, options *BaseScanOptions) (Result[string], []Result[string], error)
SScanWithOptions(key string, cursor string, options *options.BaseScanOptions) (Result[string], []Result[string], error)

// Moves `member` from the set at `source` to the set at `destination`, removing it from the source set.
// Creates a new destination set if needed. The operation is atomic.
Expand Down
Loading

0 comments on commit 8b904d4

Please sign in to comment.