gf/contrib/nosql/redis/redis_conn.go
2023-08-03 20:01:36 +08:00

277 lines
7.7 KiB
Go

// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
package redis
import (
"context"
"fmt"
"reflect"
"github.com/redis/go-redis/v9"
"go.opentelemetry.io/otel"
"go.opentelemetry.io/otel/attribute"
"go.opentelemetry.io/otel/codes"
"go.opentelemetry.io/otel/trace"
"github.com/gogf/gf/v2"
"github.com/gogf/gf/v2/container/gvar"
"github.com/gogf/gf/v2/database/gredis"
"github.com/gogf/gf/v2/encoding/gjson"
"github.com/gogf/gf/v2/errors/gerror"
"github.com/gogf/gf/v2/net/gtrace"
"github.com/gogf/gf/v2/os/gtime"
"github.com/gogf/gf/v2/text/gstr"
"github.com/gogf/gf/v2/util/gconv"
"github.com/gogf/gf/v2/util/gutil"
)
// Conn manages the connection operations.
type Conn struct {
ps *redis.PubSub
redis *Redis
}
// traceItem holds the information for redis trace.
type traceItem struct {
err error
command string
args []interface{}
costMilli int64
}
const (
traceInstrumentName = "github.com/gogf/gf/v2/database/gredis"
traceAttrRedisAddress = "redis.address"
traceAttrRedisDb = "redis.db"
traceEventRedisExecution = "redis.execution"
traceEventRedisExecutionCommand = "redis.execution.command"
traceEventRedisExecutionCost = "redis.execution.cost"
traceEventRedisExecutionArguments = "redis.execution.arguments"
)
// Do send a command to the server and returns the received reply.
// It uses json.Marshal for struct/slice/map type values before committing them to redis.
func (c *Conn) Do(ctx context.Context, command string, args ...interface{}) (reply *gvar.Var, err error) {
if ctx == nil {
ctx = context.Background()
}
for k, v := range args {
var (
reflectInfo = gutil.OriginTypeAndKind(v)
)
switch reflectInfo.OriginKind {
case
reflect.Struct,
reflect.Map,
reflect.Slice,
reflect.Array:
// Ignore slice types of: []byte.
if _, ok := v.([]byte); !ok {
if args[k], err = gjson.Marshal(v); err != nil {
return nil, err
}
}
}
}
// Trace span start.
tr := otel.GetTracerProvider().Tracer(traceInstrumentName, trace.WithInstrumentationVersion(gf.VERSION))
_, span := tr.Start(ctx, "Redis."+command, trace.WithSpanKind(trace.SpanKindInternal))
defer span.End()
timestampMilli1 := gtime.TimestampMilli()
reply, err = c.doCommand(ctx, command, args...)
timestampMilli2 := gtime.TimestampMilli()
// Trace span end.
c.traceSpanEnd(ctx, span, &traceItem{
err: err,
command: command,
args: args,
costMilli: timestampMilli2 - timestampMilli1,
})
return
}
// Do send a command to the server and returns the received reply.
// It uses json.Marshal for struct/slice/map type values before committing them to redis.
func (c *Conn) doCommand(ctx context.Context, command string, args ...interface{}) (reply *gvar.Var, err error) {
argStrSlice := gconv.Strings(args)
switch gstr.ToLower(command) {
case `subscribe`:
c.ps = c.redis.client.Subscribe(ctx, argStrSlice...)
case `psubscribe`:
c.ps = c.redis.client.PSubscribe(ctx, argStrSlice...)
case `unsubscribe`:
if c.ps != nil {
err = c.ps.Unsubscribe(ctx, argStrSlice...)
if err != nil {
err = gerror.Wrapf(err, `Redis PubSub Unsubscribe failed with arguments "%v"`, argStrSlice)
}
}
case `punsubscribe`:
if c.ps != nil {
err = c.ps.PUnsubscribe(ctx, argStrSlice...)
if err != nil {
err = gerror.Wrapf(err, `Redis PubSub PUnsubscribe failed with arguments "%v"`, argStrSlice)
}
}
default:
arguments := make([]interface{}, len(args)+1)
copy(arguments, []interface{}{command})
copy(arguments[1:], args)
reply, err = c.resultToVar(c.redis.client.Do(ctx, arguments...).Result())
if err != nil {
err = gerror.Wrapf(err, `Redis Client Do failed with arguments "%v"`, arguments)
}
}
return
}
// resultToVar converts redis operation result to gvar.Var.
func (c *Conn) resultToVar(result interface{}, err error) (*gvar.Var, error) {
if err == redis.Nil {
err = nil
}
if err == nil {
switch v := result.(type) {
case []byte:
return gvar.New(string(v)), err
case []interface{}:
return gvar.New(gconv.Strings(v)), err
case *redis.Message:
result = &gredis.Message{
Channel: v.Channel,
Pattern: v.Pattern,
Payload: v.Payload,
PayloadSlice: v.PayloadSlice,
}
case *redis.Subscription:
result = &gredis.Subscription{
Kind: v.Kind,
Channel: v.Channel,
Count: v.Count,
}
}
}
return gvar.New(result), err
}
// Receive receives a single reply as gvar.Var from the Redis server.
func (c *Conn) Receive(ctx context.Context) (*gvar.Var, error) {
if c.ps != nil {
v, err := c.resultToVar(c.ps.Receive(ctx))
if err != nil {
err = gerror.Wrapf(err, `Redis PubSub Receive failed`)
}
return v, err
}
return nil, nil
}
// Close closes current PubSub or puts the connection back to connection pool.
func (c *Conn) Close(ctx context.Context) (err error) {
if c.ps != nil {
err = c.ps.Close()
if err != nil {
err = gerror.Wrapf(err, `Redis PubSub Close failed`)
}
}
return
}
// Subscribe subscribes the client to the specified channels.
//
// https://redis.io/commands/subscribe/
func (c *Conn) Subscribe(ctx context.Context, channel string, channels ...string) ([]*gredis.Subscription, error) {
args := append([]interface{}{channel}, gconv.Interfaces(channels)...)
_, err := c.Do(ctx, "Subscribe", args...)
if err != nil {
return nil, err
}
subs := make([]*gredis.Subscription, len(args))
for i := 0; i < len(subs); i++ {
v, err := c.Receive(ctx)
if err != nil {
return nil, err
}
subs[i] = v.Val().(*gredis.Subscription)
}
return subs, err
}
// PSubscribe subscribes the client to the given patterns.
//
// Supported glob-style patterns:
// - h?llo subscribes to hello, hallo and hxllo
// - h*llo subscribes to hllo and heeeello
// - h[ae]llo subscribes to hello and hallo, but not hillo
//
// Use \ to escape special characters if you want to match them verbatim.
//
// https://redis.io/commands/psubscribe/
func (c *Conn) PSubscribe(ctx context.Context, pattern string, patterns ...string) ([]*gredis.Subscription, error) {
args := append([]interface{}{pattern}, gconv.Interfaces(patterns)...)
_, err := c.Do(ctx, "PSubscribe", args...)
if err != nil {
return nil, err
}
subs := make([]*gredis.Subscription, len(args))
for i := 0; i < len(subs); i++ {
v, err := c.Receive(ctx)
if err != nil {
return nil, err
}
subs[i] = v.Val().(*gredis.Subscription)
}
return subs, err
}
// ReceiveMessage receives a single message of subscription from the Redis server.
func (c *Conn) ReceiveMessage(ctx context.Context) (*gredis.Message, error) {
v, err := c.Receive(ctx)
if err != nil {
return nil, err
}
return v.Val().(*gredis.Message), nil
}
// traceSpanEnd checks and adds redis trace information to OpenTelemetry.
func (c *Conn) traceSpanEnd(ctx context.Context, span trace.Span, item *traceItem) {
if gtrace.IsUsingDefaultProvider() || !gtrace.IsTracingInternal() {
return
}
if ctx == nil {
ctx = context.Background()
}
if item.err != nil {
span.SetStatus(codes.Error, fmt.Sprintf(`%+v`, item.err))
}
span.SetAttributes(gtrace.CommonLabels()...)
span.SetAttributes(
attribute.String(traceAttrRedisAddress, c.redis.config.Address),
attribute.Int(traceAttrRedisDb, c.redis.config.Db),
)
jsonBytes, _ := gjson.Marshal(item.args)
span.AddEvent(traceEventRedisExecution, trace.WithAttributes(
attribute.String(traceEventRedisExecutionCommand, item.command),
attribute.String(traceEventRedisExecutionCost, fmt.Sprintf(`%d ms`, item.costMilli)),
attribute.String(traceEventRedisExecutionArguments, string(jsonBytes)),
))
}