mirror of
https://github.com/hibiken/asynq.git
synced 2024-11-15 11:58:47 +08:00
221 lines
5.2 KiB
Go
221 lines
5.2 KiB
Go
// Copyright 2020 Kentaro Hibino. All rights reserved.
|
|
// Use of this source code is governed by a MIT license
|
|
// that can be found in the LICENSE file.
|
|
|
|
package asynq
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/spf13/cast"
|
|
)
|
|
|
|
// Payload holds arbitrary data needed for task execution.
|
|
type Payload struct {
|
|
data map[string]interface{}
|
|
}
|
|
|
|
type errKeyNotFound struct {
|
|
key string
|
|
}
|
|
|
|
func (e *errKeyNotFound) Error() string {
|
|
return fmt.Sprintf("key %q does not exist", e.key)
|
|
}
|
|
|
|
// Has reports whether key exists.
|
|
func (p Payload) Has(key string) bool {
|
|
_, ok := p.data[key]
|
|
return ok
|
|
}
|
|
|
|
func toInt(v interface{}) (int, error) {
|
|
switch v := v.(type) {
|
|
case json.Number:
|
|
val, err := v.Int64()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int(val), nil
|
|
default:
|
|
return cast.ToIntE(v)
|
|
}
|
|
}
|
|
|
|
// GetString returns a string value if a string type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetString(key string) (string, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return "", &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringE(v)
|
|
}
|
|
|
|
// GetInt returns an int value if a numeric type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetInt(key string) (int, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return 0, &errKeyNotFound{key}
|
|
}
|
|
return toInt(v)
|
|
}
|
|
|
|
// GetFloat64 returns a float64 value if a numeric type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetFloat64(key string) (float64, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return 0, &errKeyNotFound{key}
|
|
}
|
|
switch v := v.(type) {
|
|
case json.Number:
|
|
return v.Float64()
|
|
default:
|
|
return cast.ToFloat64E(v)
|
|
}
|
|
}
|
|
|
|
// GetBool returns a boolean value if a boolean type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetBool(key string) (bool, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return false, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToBoolE(v)
|
|
}
|
|
|
|
// GetStringSlice returns a slice of strings if a string slice type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetStringSlice(key string) ([]string, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringSliceE(v)
|
|
}
|
|
|
|
// GetIntSlice returns a slice of ints if a int slice type is associated with
|
|
// the key, otherwise reports an error.
|
|
func (p Payload) GetIntSlice(key string) ([]int, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
switch v := v.(type) {
|
|
case []interface{}:
|
|
var res []int
|
|
for _, elem := range v {
|
|
val, err := toInt(elem)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
res = append(res, int(val))
|
|
}
|
|
return res, nil
|
|
default:
|
|
return cast.ToIntSliceE(v)
|
|
}
|
|
}
|
|
|
|
// GetStringMap returns a map of string to empty interface
|
|
// if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetStringMap(key string) (map[string]interface{}, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringMapE(v)
|
|
}
|
|
|
|
// GetStringMapString returns a map of string to string
|
|
// if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetStringMapString(key string) (map[string]string, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringMapStringE(v)
|
|
}
|
|
|
|
// GetStringMapStringSlice returns a map of string to string slice
|
|
// if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetStringMapStringSlice(key string) (map[string][]string, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringMapStringSliceE(v)
|
|
}
|
|
|
|
// GetStringMapInt returns a map of string to int
|
|
// if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetStringMapInt(key string) (map[string]int, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
switch v := v.(type) {
|
|
case map[string]interface{}:
|
|
res := make(map[string]int)
|
|
for key, val := range v {
|
|
ival, err := toInt(val)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
res[key] = ival
|
|
}
|
|
return res, nil
|
|
default:
|
|
return cast.ToStringMapIntE(v)
|
|
}
|
|
}
|
|
|
|
// GetStringMapBool returns a map of string to boolean
|
|
// if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetStringMapBool(key string) (map[string]bool, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return nil, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToStringMapBoolE(v)
|
|
}
|
|
|
|
// GetTime returns a time value if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetTime(key string) (time.Time, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return time.Time{}, &errKeyNotFound{key}
|
|
}
|
|
return cast.ToTimeE(v)
|
|
}
|
|
|
|
// GetDuration returns a duration value if a correct map type is associated with the key,
|
|
// otherwise reports an error.
|
|
func (p Payload) GetDuration(key string) (time.Duration, error) {
|
|
v, ok := p.data[key]
|
|
if !ok {
|
|
return 0, &errKeyNotFound{key}
|
|
}
|
|
switch v := v.(type) {
|
|
case json.Number:
|
|
val, err := v.Int64()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return time.Duration(val), nil
|
|
default:
|
|
return cast.ToDurationE(v)
|
|
}
|
|
}
|