You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			164 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
			
		
		
	
	
			164 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
| package redis
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| )
 | |
| 
 | |
| type BitMapCmdable interface {
 | |
| 	GetBit(ctx context.Context, key string, offset int64) *IntCmd
 | |
| 	SetBit(ctx context.Context, key string, offset int64, value int) *IntCmd
 | |
| 	BitCount(ctx context.Context, key string, bitCount *BitCount) *IntCmd
 | |
| 	BitOpAnd(ctx context.Context, destKey string, keys ...string) *IntCmd
 | |
| 	BitOpOr(ctx context.Context, destKey string, keys ...string) *IntCmd
 | |
| 	BitOpXor(ctx context.Context, destKey string, keys ...string) *IntCmd
 | |
| 	BitOpNot(ctx context.Context, destKey string, key string) *IntCmd
 | |
| 	BitPos(ctx context.Context, key string, bit int64, pos ...int64) *IntCmd
 | |
| 	BitPosSpan(ctx context.Context, key string, bit int8, start, end int64, span string) *IntCmd
 | |
| 	BitField(ctx context.Context, key string, values ...interface{}) *IntSliceCmd
 | |
| }
 | |
| 
 | |
| func (c cmdable) GetBit(ctx context.Context, key string, offset int64) *IntCmd {
 | |
| 	cmd := NewIntCmd(ctx, "getbit", key, offset)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| func (c cmdable) SetBit(ctx context.Context, key string, offset int64, value int) *IntCmd {
 | |
| 	cmd := NewIntCmd(
 | |
| 		ctx,
 | |
| 		"setbit",
 | |
| 		key,
 | |
| 		offset,
 | |
| 		value,
 | |
| 	)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| type BitCount struct {
 | |
| 	Start, End int64
 | |
| 	Unit       string // BYTE(default) | BIT
 | |
| }
 | |
| 
 | |
| const BitCountIndexByte string = "BYTE"
 | |
| const BitCountIndexBit string = "BIT"
 | |
| 
 | |
| func (c cmdable) BitCount(ctx context.Context, key string, bitCount *BitCount) *IntCmd {
 | |
| 	args := []interface{}{"bitcount", key}
 | |
| 	if bitCount != nil {
 | |
| 		if bitCount.Unit == "" {
 | |
| 			bitCount.Unit = "BYTE"
 | |
| 		}
 | |
| 		if bitCount.Unit != BitCountIndexByte && bitCount.Unit != BitCountIndexBit {
 | |
| 			cmd := NewIntCmd(ctx)
 | |
| 			cmd.SetErr(errors.New("redis: invalid bitcount index"))
 | |
| 			return cmd
 | |
| 		}
 | |
| 		args = append(
 | |
| 			args,
 | |
| 			bitCount.Start,
 | |
| 			bitCount.End,
 | |
| 			string(bitCount.Unit),
 | |
| 		)
 | |
| 	}
 | |
| 	cmd := NewIntCmd(ctx, args...)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| func (c cmdable) bitOp(ctx context.Context, op, destKey string, keys ...string) *IntCmd {
 | |
| 	args := make([]interface{}, 3+len(keys))
 | |
| 	args[0] = "bitop"
 | |
| 	args[1] = op
 | |
| 	args[2] = destKey
 | |
| 	for i, key := range keys {
 | |
| 		args[3+i] = key
 | |
| 	}
 | |
| 	cmd := NewIntCmd(ctx, args...)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| func (c cmdable) BitOpAnd(ctx context.Context, destKey string, keys ...string) *IntCmd {
 | |
| 	return c.bitOp(ctx, "and", destKey, keys...)
 | |
| }
 | |
| 
 | |
| func (c cmdable) BitOpOr(ctx context.Context, destKey string, keys ...string) *IntCmd {
 | |
| 	return c.bitOp(ctx, "or", destKey, keys...)
 | |
| }
 | |
| 
 | |
| func (c cmdable) BitOpXor(ctx context.Context, destKey string, keys ...string) *IntCmd {
 | |
| 	return c.bitOp(ctx, "xor", destKey, keys...)
 | |
| }
 | |
| 
 | |
| func (c cmdable) BitOpNot(ctx context.Context, destKey string, key string) *IntCmd {
 | |
| 	return c.bitOp(ctx, "not", destKey, key)
 | |
| }
 | |
| 
 | |
| // BitPos is an API before Redis version 7.0, cmd: bitpos key bit start end
 | |
| // if you need the `byte | bit` parameter, please use `BitPosSpan`.
 | |
| func (c cmdable) BitPos(ctx context.Context, key string, bit int64, pos ...int64) *IntCmd {
 | |
| 	args := make([]interface{}, 3+len(pos))
 | |
| 	args[0] = "bitpos"
 | |
| 	args[1] = key
 | |
| 	args[2] = bit
 | |
| 	switch len(pos) {
 | |
| 	case 0:
 | |
| 	case 1:
 | |
| 		args[3] = pos[0]
 | |
| 	case 2:
 | |
| 		args[3] = pos[0]
 | |
| 		args[4] = pos[1]
 | |
| 	default:
 | |
| 		panic("too many arguments")
 | |
| 	}
 | |
| 	cmd := NewIntCmd(ctx, args...)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| // BitPosSpan supports the `byte | bit` parameters in redis version 7.0,
 | |
| // the bitpos command defaults to using byte type for the `start-end` range,
 | |
| // which means it counts in bytes from start to end. you can set the value
 | |
| // of "span" to determine the type of `start-end`.
 | |
| // span = "bit", cmd: bitpos key bit start end bit
 | |
| // span = "byte", cmd: bitpos key bit start end byte
 | |
| func (c cmdable) BitPosSpan(ctx context.Context, key string, bit int8, start, end int64, span string) *IntCmd {
 | |
| 	cmd := NewIntCmd(ctx, "bitpos", key, bit, start, end, span)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| // BitField accepts multiple values:
 | |
| //   - BitField("set", "i1", "offset1", "value1","cmd2", "type2", "offset2", "value2")
 | |
| //   - BitField([]string{"cmd1", "type1", "offset1", "value1","cmd2", "type2", "offset2", "value2"})
 | |
| //   - BitField([]interface{}{"cmd1", "type1", "offset1", "value1","cmd2", "type2", "offset2", "value2"})
 | |
| func (c cmdable) BitField(ctx context.Context, key string, values ...interface{}) *IntSliceCmd {
 | |
| 	args := make([]interface{}, 2, 2+len(values))
 | |
| 	args[0] = "bitfield"
 | |
| 	args[1] = key
 | |
| 	args = appendArgs(args, values)
 | |
| 	cmd := NewIntSliceCmd(ctx, args...)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| // BitFieldRO - Read-only variant of the BITFIELD command.
 | |
| // It is like the original BITFIELD but only accepts GET subcommand and can safely be used in read-only replicas.
 | |
| // - BitFieldRO(ctx, key, "<Encoding0>", "<Offset0>", "<Encoding1>","<Offset1>")
 | |
| func (c cmdable) BitFieldRO(ctx context.Context, key string, values ...interface{}) *IntSliceCmd {
 | |
| 	args := make([]interface{}, 2, 2+len(values))
 | |
| 	args[0] = "BITFIELD_RO"
 | |
| 	args[1] = key
 | |
| 	if len(values)%2 != 0 {
 | |
| 		panic("BitFieldRO: invalid number of arguments, must be even")
 | |
| 	}
 | |
| 	for i := 0; i < len(values); i += 2 {
 | |
| 		args = append(args, "GET", values[i], values[i+1])
 | |
| 	}
 | |
| 	cmd := NewIntSliceCmd(ctx, args...)
 | |
| 	_ = c(ctx, cmd)
 | |
| 	return cmd
 | |
| }
 |