mirror of
https://github.com/hibiken/asynq.git
synced 2024-11-10 11:31:58 +08:00
Use md5 to generate checksum for unique key
This commit is contained in:
parent
68dd6d9a9d
commit
a0df047f71
@ -7,6 +7,8 @@ package base
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/md5"
|
||||||
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
@ -170,9 +172,12 @@ func SchedulerHistoryKey(entryID string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// UniqueKey returns a redis key with the given type, payload, and queue name.
|
// UniqueKey returns a redis key with the given type, payload, and queue name.
|
||||||
// FIXME: We probably need to generate a hash of payload to make this key unique
|
|
||||||
func UniqueKey(qname, tasktype string, payload []byte) string {
|
func UniqueKey(qname, tasktype string, payload []byte) string {
|
||||||
return fmt.Sprintf("%sunique:%s:%s", QueueKeyPrefix(qname), tasktype, string(payload))
|
if payload == nil {
|
||||||
|
return fmt.Sprintf("%sunique:%s:", QueueKeyPrefix(qname), tasktype)
|
||||||
|
}
|
||||||
|
checksum := md5.Sum(payload)
|
||||||
|
return fmt.Sprintf("%sunique:%s:%s", QueueKeyPrefix(qname), tasktype, hex.EncodeToString(checksum[:]))
|
||||||
}
|
}
|
||||||
|
|
||||||
// TaskMessage is the internal representation of a task with additional metadata fields.
|
// TaskMessage is the internal representation of a task with additional metadata fields.
|
||||||
|
@ -6,6 +6,8 @@ package base
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/md5"
|
||||||
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
@ -276,6 +278,19 @@ func toBytes(m map[string]interface{}) []byte {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestUniqueKey(t *testing.T) {
|
func TestUniqueKey(t *testing.T) {
|
||||||
|
payload1 := toBytes(map[string]interface{}{"a": 123, "b": "hello", "c": true})
|
||||||
|
payload2 := toBytes(map[string]interface{}{"b": "hello", "c": true, "a": 123})
|
||||||
|
payload3 := toBytes(map[string]interface{}{
|
||||||
|
"address": map[string]string{"line": "123 Main St", "city": "Boston", "state": "MA"},
|
||||||
|
"names": []string{"bob", "mike", "rob"}})
|
||||||
|
payload4 := toBytes(map[string]interface{}{
|
||||||
|
"time": time.Date(2020, time.July, 28, 0, 0, 0, 0, time.UTC),
|
||||||
|
"duration": time.Hour})
|
||||||
|
|
||||||
|
checksum := func(data []byte) string {
|
||||||
|
sum := md5.Sum(data)
|
||||||
|
return hex.EncodeToString(sum[:])
|
||||||
|
}
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
desc string
|
desc string
|
||||||
qname string
|
qname string
|
||||||
@ -287,41 +302,29 @@ func TestUniqueKey(t *testing.T) {
|
|||||||
"with primitive types",
|
"with primitive types",
|
||||||
"default",
|
"default",
|
||||||
"email:send",
|
"email:send",
|
||||||
toBytes(map[string]interface{}{"a": 123, "b": "hello", "c": true}),
|
payload1,
|
||||||
fmt.Sprintf("asynq:{default}:unique:email:send:%s",
|
fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload1)),
|
||||||
string(toBytes(map[string]interface{}{"a": 123, "b": "hello", "c": true}))),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"with unsorted keys",
|
"with unsorted keys",
|
||||||
"default",
|
"default",
|
||||||
"email:send",
|
"email:send",
|
||||||
toBytes(map[string]interface{}{"b": "hello", "c": true, "a": 123}),
|
payload2,
|
||||||
fmt.Sprintf("asynq:{default}:unique:email:send:%s",
|
fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload2)),
|
||||||
string(toBytes(map[string]interface{}{"b": "hello", "c": true, "a": 123}))),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"with composite types",
|
"with composite types",
|
||||||
"default",
|
"default",
|
||||||
"email:send",
|
"email:send",
|
||||||
toBytes(map[string]interface{}{
|
payload3,
|
||||||
"address": map[string]string{"line": "123 Main St", "city": "Boston", "state": "MA"},
|
fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload3)),
|
||||||
"names": []string{"bob", "mike", "rob"}}),
|
|
||||||
fmt.Sprintf("asynq:{default}:unique:email:send:%s",
|
|
||||||
string(toBytes(map[string]interface{}{
|
|
||||||
"address": map[string]string{"line": "123 Main St", "city": "Boston", "state": "MA"},
|
|
||||||
"names": []string{"bob", "mike", "rob"}}))),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"with complex types",
|
"with complex types",
|
||||||
"default",
|
"default",
|
||||||
"email:send",
|
"email:send",
|
||||||
toBytes(map[string]interface{}{
|
payload4,
|
||||||
"time": time.Date(2020, time.July, 28, 0, 0, 0, 0, time.UTC),
|
fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload4)),
|
||||||
"duration": time.Hour}),
|
|
||||||
fmt.Sprintf("asynq:{default}:unique:email:send:%s",
|
|
||||||
string(toBytes(map[string]interface{}{
|
|
||||||
"time": time.Date(2020, time.July, 28, 0, 0, 0, 0, time.UTC),
|
|
||||||
"duration": time.Hour}))),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"with nil payload",
|
"with nil payload",
|
||||||
|
Loading…
Reference in New Issue
Block a user