mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-20 21:26:14 +08:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			develop
			...
			revert-491
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | fd08c06232 | 
							
								
								
									
										21
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +0,0 @@ | ||||
| version: 2 | ||||
| updates: | ||||
|   - package-ecosystem: "gomod" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     labels: | ||||
|       - "pr-deps" | ||||
|   - package-ecosystem: "gomod" | ||||
|     directory: "/tools" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     labels: | ||||
|       - "pr-deps" | ||||
|   - package-ecosystem: "gomod" | ||||
|     directory: "/x" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     labels: | ||||
|       - "pr-deps" | ||||
|        | ||||
							
								
								
									
										32
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,20 +11,20 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis:7 | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v4 | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.21.x | ||||
|           go-version: 1.16.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a new.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|           path: new.txt | ||||
| @@ -33,22 +33,22 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis:7 | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           ref: master | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v4 | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.21.x | ||||
|           go-version: 1.15.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a old.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: bench-current | ||||
|           path: old.txt | ||||
| @@ -58,25 +58,25 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v4 | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.21.x | ||||
|           go-version: 1.15.x | ||||
|       - name: Install benchstat | ||||
|         run: go get -u golang.org/x/perf/cmd/benchstat | ||||
|       - name: Download Incoming | ||||
|         uses: actions/download-artifact@v3 | ||||
|         uses: actions/download-artifact@v2 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|       - name: Download Current | ||||
|         uses: actions/download-artifact@v3 | ||||
|         uses: actions/download-artifact@v2 | ||||
|         with: | ||||
|           name: bench-current | ||||
|       - name: Benchstat Results | ||||
|         run: benchstat old.txt new.txt | tee -a benchstat.txt | ||||
|       - name: Upload benchstat results | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: benchstat | ||||
|           path: benchstat.txt | ||||
|   | ||||
							
								
								
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,21 +7,20 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         go-version: [1.20.x, 1.21.x] | ||||
|         go-version: [1.14.x, 1.15.x, 1.16.x, 1.17.x, 1.18.x] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis:7 | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v2 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v4 | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go-version }} | ||||
|           cache: false | ||||
|  | ||||
|       - name: Build core module | ||||
|         run: go build -v ./... | ||||
| @@ -45,21 +44,20 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         go-version: [1.20.x, 1.21.x] | ||||
|         go-version: [1.18.x] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis:7 | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v2 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v4 | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go-version }} | ||||
|           cache: false | ||||
|  | ||||
|       - name: Build tools module | ||||
|         run: cd tools && go build -v ./... && cd .. | ||||
|   | ||||
| @@ -7,11 +7,6 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | ||||
|  | ||||
| ## [Unreleased] | ||||
|  | ||||
| ## [0.25.0] - 2023-01-02 | ||||
|  | ||||
| ### Added | ||||
| - Added configuration for Janitor's Interval and Deletion Batch Size (PR: https://github.com/hibiken/asynq/pull/715) | ||||
|  | ||||
| ## [0.24.1] - 2023-05-01 | ||||
|  | ||||
| ### Changed | ||||
|   | ||||
| @@ -54,7 +54,7 @@ If you are using this package in production, **please consider sponsoring the pr | ||||
|  | ||||
| ## Quickstart | ||||
|  | ||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). Latest two Go versions are supported (See https://go.dev/dl). | ||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). Version `1.14` or higher is required. | ||||
|  | ||||
| Initialize your project by creating a folder and then running `go mod init github.com/your/repo` ([learn more](https://blog.golang.org/using-go-modules)) inside the folder. Then install Asynq library with the [`go get`](https://golang.org/cmd/go/#hdr-Add_dependencies_to_current_module_and_install_them) command: | ||||
|  | ||||
| @@ -295,7 +295,7 @@ Asynq ships with a command line tool to inspect the state of queues and tasks. | ||||
| To install the CLI tool, run the following command: | ||||
|  | ||||
| ```sh | ||||
| go install github.com/hibiken/asynq/tools/asynq@latest | ||||
| go install github.com/hibiken/asynq/tools/asynq | ||||
| ``` | ||||
|  | ||||
| Here's an example of running the `asynq dash` command: | ||||
|   | ||||
							
								
								
									
										26
									
								
								client.go
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								client.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // A Client is responsible for scheduling tasks. | ||||
| @@ -25,26 +25,15 @@ import ( | ||||
| // Clients are safe for concurrent use by multiple goroutines. | ||||
| type Client struct { | ||||
| 	broker base.Broker | ||||
| 	// When a Client has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
| } | ||||
|  | ||||
| // NewClient returns a new Client instance given a redis connection option. | ||||
| func NewClient(r RedisConnOpt) *Client { | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	client := NewClientFromRedisClient(redisClient) | ||||
| 	client.sharedConnection = false | ||||
| 	return client | ||||
| } | ||||
|  | ||||
| // NewClientFromRedisClient returns a new instance of Client given a redis.UniversalClient | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewClientFromRedisClient(c redis.UniversalClient) *Client { | ||||
| 	return &Client{broker: rdb.NewRDB(c), sharedConnection: true} | ||||
| 	return &Client{broker: rdb.NewRDB(c)} | ||||
| } | ||||
|  | ||||
| type OptionType int | ||||
| @@ -161,9 +150,9 @@ func (t deadlineOption) Value() interface{} { return time.Time(t) } | ||||
| // TTL duration must be greater than or equal to 1 second. | ||||
| // | ||||
| // Uniqueness of a task is based on the following properties: | ||||
| //   - Task Type | ||||
| //   - Task Payload | ||||
| //   - Queue Name | ||||
| //     - Task Type | ||||
| //     - Task Payload | ||||
| //     - Queue Name | ||||
| func Unique(ttl time.Duration) Option { | ||||
| 	return uniqueOption(ttl) | ||||
| } | ||||
| @@ -318,9 +307,6 @@ var ( | ||||
|  | ||||
| // Close closes the connection with redis. | ||||
| func (c *Client) Close() error { | ||||
| 	if c.sharedConnection { | ||||
| 		return fmt.Errorf("redis connection is shared so the Client can't be closed through asynq") | ||||
| 	} | ||||
| 	return c.broker.Close() | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -14,7 +14,6 @@ import ( | ||||
| 	"github.com/google/go-cmp/cmp/cmpopts" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	h "github.com/hibiken/asynq/internal/testutil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| @@ -144,7 +143,11 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func testClientEnqueue(t *testing.T, client *Client, r redis.UniversalClient) { | ||||
| func TestClientEnqueue(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
| 	now := time.Now() | ||||
|  | ||||
| @@ -475,24 +478,6 @@ func testClientEnqueue(t *testing.T, client *Client, r redis.UniversalClient) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueue(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
| 	testClientEnqueue(t, client, r) | ||||
| } | ||||
|  | ||||
| func TestClientFromRedisClientEnqueue(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	redisClient := getRedisConnOpt(t).MakeRedisClient().(redis.UniversalClient) | ||||
| 	client := NewClientFromRedisClient(redisClient) | ||||
| 	testClientEnqueue(t, client, r) | ||||
| 	err := client.Close() | ||||
| 	if err == nil { | ||||
| 		t.Error("client.Close() should have failed because of a shared client but it didn't") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueueWithGroupOption(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
|   | ||||
							
								
								
									
										26
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,21 +1,17 @@ | ||||
| module github.com/hibiken/asynq | ||||
|  | ||||
| go 1.20 | ||||
| go 1.14 | ||||
|  | ||||
| require ( | ||||
| 	github.com/golang/protobuf v1.5.3 | ||||
| 	github.com/google/go-cmp v0.6.0 | ||||
| 	github.com/google/uuid v1.6.0 | ||||
| 	github.com/redis/go-redis/v9 v9.4.0 | ||||
| 	github.com/golang/protobuf v1.5.2 | ||||
| 	github.com/google/go-cmp v0.5.6 | ||||
| 	github.com/google/uuid v1.2.0 | ||||
| 	github.com/redis/go-redis/v9 v9.0.3 | ||||
| 	github.com/robfig/cron/v3 v3.0.1 | ||||
| 	github.com/spf13/cast v1.5.1 | ||||
| 	go.uber.org/goleak v1.3.0 | ||||
| 	golang.org/x/sys v0.16.0 | ||||
| 	golang.org/x/time v0.3.0 | ||||
| 	google.golang.org/protobuf v1.31.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/spf13/cast v1.3.1 | ||||
| 	go.uber.org/goleak v1.1.12 | ||||
| 	golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e | ||||
| 	golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 | ||||
| 	google.golang.org/protobuf v1.26.0 | ||||
| 	gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										95
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,39 +1,82 @@ | ||||
| github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/bsm/ginkgo/v2 v2.7.0 h1:ItPMPH90RbmZJt5GtkcNvIRuGEdwlBItdNVoyzaNQao= | ||||
| github.com/bsm/ginkgo/v2 v2.7.0/go.mod h1:AiKlXPm7ItEHNc/2+OkrNG4E0ITzojb9/xWzvQ9XZ9w= | ||||
| github.com/bsm/gomega v1.26.0 h1:LhQm+AFcgV2M0WyKroMASzAzCAJVpAxQXv4SaI9a69Y= | ||||
| github.com/bsm/gomega v1.26.0/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0XL9UY= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= | ||||
| github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/google/go-cmp v0.5.6 h1:BKbKCqvP6I+rmFHt06ZmyQtvB8xAkWdhFyr0ZUNZcxQ= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/uuid v1.2.0 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/redis/go-redis/v9 v9.4.0 h1:Yzoz33UZw9I/mFhx4MNrB6Fk+XHO1VukNcCa1+lwyKk= | ||||
| github.com/redis/go-redis/v9 v9.4.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/redis/go-redis/v9 v9.0.3 h1:+7mmR26M0IvyLxGZUHxu4GiBkJkVDid0Un+j4ScYu4k= | ||||
| github.com/redis/go-redis/v9 v9.0.3/go.mod h1:WqMKv5vnQbRuZstUwxQI195wHy+t4PuXDOjzMvcuQHk= | ||||
| github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= | ||||
| github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | ||||
| github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||||
| github.com/spf13/cast v1.5.1 h1:R+kOtfhWQE6TVQzY+4D7wJLBgkdVasCEFxSUBYBYIlA= | ||||
| github.com/spf13/cast v1.5.1/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48= | ||||
| github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= | ||||
| go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||
| go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||
| golang.org/x/sys v0.16.0 h1:xWw16ngr6ZMtmxDyKyIgsE93KNKz5HKmMa3b8ALHidU= | ||||
| golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= | ||||
| golang.org/x/time v0.3.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| github.com/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= | ||||
| github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||
| go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= | ||||
| go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de h1:5hukYrvBGR8/eNkX5mdUezrA6JiaEZDtJb9Ei+1LlBs= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.4.2 h1:Gz96sIWK3OalVv/I/qNygP42zyoKp3xptRVCWRFEBvo= | ||||
| golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e h1:fLOSk5Q00efkSvAm+4xcoXD+RRmLmmulPn5I3Y9F2EM= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.1.5 h1:ouewzE6p+/VEB31YYnTbEJdi8pFqKp4P4n85vwo3DHA= | ||||
| golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8= | ||||
| google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
|   | ||||
							
								
								
									
										19
									
								
								inspector.go
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								inspector.go
									
									
									
									
									
								
							| @@ -10,19 +10,16 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // Inspector is a client interface to inspect and mutate the state of | ||||
| // queues and tasks. | ||||
| type Inspector struct { | ||||
| 	rdb *rdb.RDB | ||||
| 	// When an Inspector has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
| } | ||||
|  | ||||
| // New returns a new instance of Inspector. | ||||
| @@ -31,25 +28,13 @@ func NewInspector(r RedisConnOpt) *Inspector { | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("inspeq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	inspector := NewInspectorFromRedisClient(c) | ||||
| 	inspector.sharedConnection = false | ||||
| 	return inspector | ||||
| } | ||||
|  | ||||
| // NewInspectorFromRedisClient returns a new instance of Inspector given a redis.UniversalClient | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewInspectorFromRedisClient(c redis.UniversalClient) *Inspector { | ||||
| 	return &Inspector{ | ||||
| 		rdb:              rdb.NewRDB(c), | ||||
| 		sharedConnection: true, | ||||
| 		rdb: rdb.NewRDB(c), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Close closes the connection with redis. | ||||
| func (i *Inspector) Close() error { | ||||
| 	if i.sharedConnection { | ||||
| 		return fmt.Errorf("redis connection is shared so the Inspector can't be closed through asynq") | ||||
| 	} | ||||
| 	return i.rdb.Close() | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -22,7 +22,11 @@ import ( | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| func testInspectorQueues(t *testing.T, inspector *Inspector, r redis.UniversalClient) { | ||||
| func TestInspectorQueues(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	inspector := NewInspector(getRedisConnOpt(t)) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		queues []string | ||||
| 	}{ | ||||
| @@ -48,21 +52,7 @@ func testInspectorQueues(t *testing.T, inspector *Inspector, r redis.UniversalCl | ||||
| 			t.Errorf("Queues() = %v, want %v; (-want, +got)\n%s", got, tc.queues, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestInspectorQueues(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	inspector := NewInspector(getRedisConnOpt(t)) | ||||
| 	testInspectorQueues(t, inspector, r) | ||||
| } | ||||
|  | ||||
| func TestInspectorFromRedisClientQueues(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	redisClient := getRedisConnOpt(t).MakeRedisClient().(redis.UniversalClient) | ||||
| 	inspector := NewInspectorFromRedisClient(redisClient) | ||||
| 	testInspectorQueues(t, inspector, r) | ||||
| } | ||||
|  | ||||
| func TestInspectorDeleteQueue(t *testing.T) { | ||||
|   | ||||
| @@ -737,7 +737,7 @@ type Broker interface { | ||||
| 	ReclaimStaleAggregationSets(qname string) error | ||||
|  | ||||
| 	// Task retention related method | ||||
| 	DeleteExpiredCompletedTasks(qname string, batchSize int) error | ||||
| 	DeleteExpiredCompletedTasks(qname string) error | ||||
|  | ||||
| 	// Lease related methods | ||||
| 	ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*TaskMessage, error) | ||||
|   | ||||
| @@ -256,21 +256,6 @@ func IsRedisCommandError(err error) bool { | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // PanicError defines an error when occurred a panic error. | ||||
| type PanicError struct { | ||||
| 	ErrMsg string | ||||
| } | ||||
|  | ||||
| func (e *PanicError) Error() string { | ||||
| 	return fmt.Sprintf("panic error cause by: %s", e.ErrMsg) | ||||
| } | ||||
|  | ||||
| // IsPanicError reports whether any error in err's chain is of type PanicError. | ||||
| func IsPanicError(err error) bool { | ||||
| 	var target *PanicError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| /************************************************* | ||||
|     Standard Library errors package functions | ||||
| *************************************************/ | ||||
|   | ||||
| @@ -131,12 +131,6 @@ func TestErrorPredicates(t *testing.T) { | ||||
| 			err:  E(Op("rdb.ArchiveTask"), NotFound, &QueueNotFoundError{Queue: "default"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "IsPanicError should detect presence of PanicError in err's chain", | ||||
| 			fn:   IsPanicError, | ||||
| 			err:  E(Op("unknown"), Unknown, &PanicError{ErrMsg: "Something went wrong"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
|   | ||||
| @@ -1217,7 +1217,7 @@ redis.call("ZREMRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | ||||
| return redis.status_reply("OK") | ||||
| `) | ||||
|  | ||||
| // ReclaimStaleAggregationSets checks for any stale aggregation sets in the given queue, and | ||||
| // ReclaimStateAggregationSets checks for any stale aggregation sets in the given queue, and | ||||
| // reclaim tasks in the stale aggregation set by putting them back in the group. | ||||
| func (r *RDB) ReclaimStaleAggregationSets(qname string) error { | ||||
| 	var op errors.Op = "RDB.ReclaimStaleAggregationSets" | ||||
| @@ -1241,7 +1241,9 @@ return table.getn(ids)`) | ||||
|  | ||||
| // DeleteExpiredCompletedTasks checks for any expired tasks in the given queue's completed set, | ||||
| // and delete all expired tasks. | ||||
| func (r *RDB) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||
| func (r *RDB) DeleteExpiredCompletedTasks(qname string) error { | ||||
| 	// Note: Do this operation in fix batches to prevent long running script. | ||||
| 	const batchSize = 100 | ||||
| 	for { | ||||
| 		n, err := r.deleteExpiredCompletedTasks(qname, batchSize) | ||||
| 		if err != nil { | ||||
| @@ -1282,10 +1284,7 @@ local res = {} | ||||
| local ids = redis.call("ZRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | ||||
| for _, id in ipairs(ids) do | ||||
| 	local key = ARGV[2] .. id | ||||
| 	local v = redis.call("HGET", key, "msg") | ||||
| 	if v then | ||||
| 		table.insert(res, v) | ||||
| 	end | ||||
| 	table.insert(res, redis.call("HGET", key, "msg")) | ||||
| end | ||||
| return res | ||||
| `) | ||||
|   | ||||
| @@ -2542,8 +2542,8 @@ func TestDeleteExpiredCompletedTasks(t *testing.T) { | ||||
| 		h.FlushDB(t, r.client) | ||||
| 		h.SeedAllCompletedQueues(t, r.client, tc.completed) | ||||
|  | ||||
| 		if err := r.DeleteExpiredCompletedTasks(tc.qname, 100); err != nil { | ||||
| 			t.Errorf("DeleteExpiredCompletedTasks(%q, 100) failed: %v", tc.qname, err) | ||||
| 		if err := r.DeleteExpiredCompletedTasks(tc.qname); err != nil { | ||||
| 			t.Errorf("DeleteExpiredCompletedTasks(%q) failed: %v", tc.qname, err) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
|   | ||||
| @@ -11,8 +11,8 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| var errRedisDown = errors.New("testutil: redis is down") | ||||
| @@ -145,13 +145,13 @@ func (tb *TestBroker) ForwardIfReady(qnames ...string) error { | ||||
| 	return tb.real.ForwardIfReady(qnames...) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return errRedisDown | ||||
| 	} | ||||
| 	return tb.real.DeleteExpiredCompletedTasks(qname, batchSize) | ||||
| 	return tb.real.DeleteExpiredCompletedTasks(qname) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*base.TaskMessage, error) { | ||||
|   | ||||
							
								
								
									
										15
									
								
								janitor.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								janitor.go
									
									
									
									
									
								
							| @@ -27,17 +27,13 @@ type janitor struct { | ||||
|  | ||||
| 	// average interval between checks. | ||||
| 	avgInterval time.Duration | ||||
|  | ||||
| 	// number of tasks to be deleted when janitor runs to delete the expired completed tasks. | ||||
| 	batchSize int | ||||
| } | ||||
|  | ||||
| type janitorParams struct { | ||||
| 	logger    *log.Logger | ||||
| 	broker    base.Broker | ||||
| 	queues    []string | ||||
| 	interval  time.Duration | ||||
| 	batchSize int | ||||
| 	logger   *log.Logger | ||||
| 	broker   base.Broker | ||||
| 	queues   []string | ||||
| 	interval time.Duration | ||||
| } | ||||
|  | ||||
| func newJanitor(params janitorParams) *janitor { | ||||
| @@ -47,7 +43,6 @@ func newJanitor(params janitorParams) *janitor { | ||||
| 		done:        make(chan struct{}), | ||||
| 		queues:      params.queues, | ||||
| 		avgInterval: params.interval, | ||||
| 		batchSize:   params.batchSize, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -78,7 +73,7 @@ func (j *janitor) start(wg *sync.WaitGroup) { | ||||
|  | ||||
| func (j *janitor) exec() { | ||||
| 	for _, qname := range j.queues { | ||||
| 		if err := j.broker.DeleteExpiredCompletedTasks(qname, j.batchSize); err != nil { | ||||
| 		if err := j.broker.DeleteExpiredCompletedTasks(qname); err != nil { | ||||
| 			j.logger.Errorf("Failed to delete expired completed tasks from queue %q: %v", | ||||
| 				qname, err) | ||||
| 		} | ||||
|   | ||||
| @@ -26,13 +26,11 @@ func TestJanitor(t *testing.T) { | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
| 	const interval = 1 * time.Second | ||||
| 	const batchSize = 100 | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:    testLogger, | ||||
| 		broker:    rdbClient, | ||||
| 		queues:    []string{"default", "custom"}, | ||||
| 		interval:  interval, | ||||
| 		batchSize: batchSize, | ||||
| 		logger:   testLogger, | ||||
| 		broker:   rdbClient, | ||||
| 		queues:   []string{"default", "custom"}, | ||||
| 		interval: interval, | ||||
| 	}) | ||||
|  | ||||
| 	now := time.Now() | ||||
|   | ||||
							
								
								
									
										93
									
								
								processor.go
									
									
									
									
									
								
							
							
						
						
									
										93
									
								
								processor.go
									
									
									
									
									
								
							| @@ -37,11 +37,11 @@ type processor struct { | ||||
| 	// orderedQueues is set only in strict-priority mode. | ||||
| 	orderedQueues []string | ||||
|  | ||||
| 	taskCheckInterval time.Duration | ||||
| 	retryDelayFunc    RetryDelayFunc | ||||
| 	isFailureFunc     func(error) bool | ||||
| 	retryDelayFunc RetryDelayFunc | ||||
| 	isFailureFunc  func(error) bool | ||||
|  | ||||
| 	errHandler ErrorHandler | ||||
|  | ||||
| 	errHandler      ErrorHandler | ||||
| 	shutdownTimeout time.Duration | ||||
|  | ||||
| 	// channel via which to send sync requests to syncer. | ||||
| @@ -73,21 +73,20 @@ type processor struct { | ||||
| } | ||||
|  | ||||
| type processorParams struct { | ||||
| 	logger            *log.Logger | ||||
| 	broker            base.Broker | ||||
| 	baseCtxFn         func() context.Context | ||||
| 	retryDelayFunc    RetryDelayFunc | ||||
| 	taskCheckInterval time.Duration | ||||
| 	isFailureFunc     func(error) bool | ||||
| 	syncCh            chan<- *syncRequest | ||||
| 	cancelations      *base.Cancelations | ||||
| 	concurrency       int | ||||
| 	queues            map[string]int | ||||
| 	strictPriority    bool | ||||
| 	errHandler        ErrorHandler | ||||
| 	shutdownTimeout   time.Duration | ||||
| 	starting          chan<- *workerInfo | ||||
| 	finished          chan<- *base.TaskMessage | ||||
| 	logger          *log.Logger | ||||
| 	broker          base.Broker | ||||
| 	baseCtxFn       func() context.Context | ||||
| 	retryDelayFunc  RetryDelayFunc | ||||
| 	isFailureFunc   func(error) bool | ||||
| 	syncCh          chan<- *syncRequest | ||||
| 	cancelations    *base.Cancelations | ||||
| 	concurrency     int | ||||
| 	queues          map[string]int | ||||
| 	strictPriority  bool | ||||
| 	errHandler      ErrorHandler | ||||
| 	shutdownTimeout time.Duration | ||||
| 	starting        chan<- *workerInfo | ||||
| 	finished        chan<- *base.TaskMessage | ||||
| } | ||||
|  | ||||
| // newProcessor constructs a new processor. | ||||
| @@ -98,27 +97,26 @@ func newProcessor(params processorParams) *processor { | ||||
| 		orderedQueues = sortByPriority(queues) | ||||
| 	} | ||||
| 	return &processor{ | ||||
| 		logger:            params.logger, | ||||
| 		broker:            params.broker, | ||||
| 		baseCtxFn:         params.baseCtxFn, | ||||
| 		clock:             timeutil.NewRealClock(), | ||||
| 		queueConfig:       queues, | ||||
| 		orderedQueues:     orderedQueues, | ||||
| 		taskCheckInterval: params.taskCheckInterval, | ||||
| 		retryDelayFunc:    params.retryDelayFunc, | ||||
| 		isFailureFunc:     params.isFailureFunc, | ||||
| 		syncRequestCh:     params.syncCh, | ||||
| 		cancelations:      params.cancelations, | ||||
| 		errLogLimiter:     rate.NewLimiter(rate.Every(3*time.Second), 1), | ||||
| 		sema:              make(chan struct{}, params.concurrency), | ||||
| 		done:              make(chan struct{}), | ||||
| 		quit:              make(chan struct{}), | ||||
| 		abort:             make(chan struct{}), | ||||
| 		errHandler:        params.errHandler, | ||||
| 		handler:           HandlerFunc(func(ctx context.Context, t *Task) error { return fmt.Errorf("handler not set") }), | ||||
| 		shutdownTimeout:   params.shutdownTimeout, | ||||
| 		starting:          params.starting, | ||||
| 		finished:          params.finished, | ||||
| 		logger:          params.logger, | ||||
| 		broker:          params.broker, | ||||
| 		baseCtxFn:       params.baseCtxFn, | ||||
| 		clock:           timeutil.NewRealClock(), | ||||
| 		queueConfig:     queues, | ||||
| 		orderedQueues:   orderedQueues, | ||||
| 		retryDelayFunc:  params.retryDelayFunc, | ||||
| 		isFailureFunc:   params.isFailureFunc, | ||||
| 		syncRequestCh:   params.syncCh, | ||||
| 		cancelations:    params.cancelations, | ||||
| 		errLogLimiter:   rate.NewLimiter(rate.Every(3*time.Second), 1), | ||||
| 		sema:            make(chan struct{}, params.concurrency), | ||||
| 		done:            make(chan struct{}), | ||||
| 		quit:            make(chan struct{}), | ||||
| 		abort:           make(chan struct{}), | ||||
| 		errHandler:      params.errHandler, | ||||
| 		handler:         HandlerFunc(func(ctx context.Context, t *Task) error { return fmt.Errorf("handler not set") }), | ||||
| 		shutdownTimeout: params.shutdownTimeout, | ||||
| 		starting:        params.starting, | ||||
| 		finished:        params.finished, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -181,7 +179,7 @@ func (p *processor) exec() { | ||||
| 			// Sleep to avoid slamming redis and let scheduler move tasks into queues. | ||||
| 			// Note: We are not using blocking pop operation and polling queues instead. | ||||
| 			// This adds significant load to redis. | ||||
| 			time.Sleep(p.taskCheckInterval) | ||||
| 			time.Sleep(time.Second) | ||||
| 			<-p.sema // release token | ||||
| 			return | ||||
| 		case err != nil: | ||||
| @@ -422,15 +420,12 @@ func (p *processor) perform(ctx context.Context, task *Task) (err error) { | ||||
| 				// map/slice usage. The parent frame should have the real trigger. | ||||
| 				_, file, line, ok = runtime.Caller(2) | ||||
| 			} | ||||
| 			var errMsg string | ||||
|  | ||||
| 			// Include the file and line number info in the error, if runtime.Caller returned ok. | ||||
| 			if ok { | ||||
| 				errMsg = fmt.Sprintf("panic [%s:%d]: %v", file, line, x) | ||||
| 				err = fmt.Errorf("panic [%s:%d]: %v", file, line, x) | ||||
| 			} else { | ||||
| 				errMsg = fmt.Sprintf("panic: %v", x) | ||||
| 			} | ||||
| 			err = &errors.PanicError{ | ||||
| 				ErrMsg: errMsg, | ||||
| 				err = fmt.Errorf("panic: %v", x) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| @@ -526,7 +521,3 @@ func (p *processor) computeDeadline(msg *base.TaskMessage) time.Time { | ||||
| 	} | ||||
| 	return time.Unix(msg.Deadline, 0) | ||||
| } | ||||
|  | ||||
| func IsPanicError(err error) bool { | ||||
| 	return errors.IsPanicError(err) | ||||
| } | ||||
|   | ||||
| @@ -9,7 +9,6 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @@ -63,21 +62,20 @@ func newProcessorForTest(t *testing.T, r *rdb.RDB, h Handler) *processor { | ||||
| 	go fakeHeartbeater(starting, finished, done) | ||||
| 	go fakeSyncer(syncCh, done) | ||||
| 	p := newProcessor(processorParams{ | ||||
| 		logger:            testLogger, | ||||
| 		broker:            r, | ||||
| 		baseCtxFn:         context.Background, | ||||
| 		retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 		taskCheckInterval: defaultTaskCheckInterval, | ||||
| 		isFailureFunc:     defaultIsFailureFunc, | ||||
| 		syncCh:            syncCh, | ||||
| 		cancelations:      base.NewCancelations(), | ||||
| 		concurrency:       10, | ||||
| 		queues:            defaultQueueConfig, | ||||
| 		strictPriority:    false, | ||||
| 		errHandler:        nil, | ||||
| 		shutdownTimeout:   defaultShutdownTimeout, | ||||
| 		starting:          starting, | ||||
| 		finished:          finished, | ||||
| 		logger:          testLogger, | ||||
| 		broker:          r, | ||||
| 		baseCtxFn:       context.Background, | ||||
| 		retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 		isFailureFunc:   defaultIsFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    base.NewCancelations(), | ||||
| 		concurrency:     10, | ||||
| 		queues:          defaultQueueConfig, | ||||
| 		strictPriority:  false, | ||||
| 		errHandler:      nil, | ||||
| 		shutdownTimeout: defaultShutdownTimeout, | ||||
| 		starting:        starting, | ||||
| 		finished:        finished, | ||||
| 	}) | ||||
| 	p.handler = h | ||||
| 	return p | ||||
| @@ -540,21 +538,20 @@ func TestProcessorWithExpiredLease(t *testing.T) { | ||||
| 		}() | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:            testLogger, | ||||
| 			broker:            rdbClient, | ||||
| 			baseCtxFn:         context.Background, | ||||
| 			taskCheckInterval: defaultTaskCheckInterval, | ||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:     defaultIsFailureFunc, | ||||
| 			syncCh:            syncCh, | ||||
| 			cancelations:      base.NewCancelations(), | ||||
| 			concurrency:       10, | ||||
| 			queues:            defaultQueueConfig, | ||||
| 			strictPriority:    false, | ||||
| 			errHandler:        nil, | ||||
| 			shutdownTimeout:   defaultShutdownTimeout, | ||||
| 			starting:          starting, | ||||
| 			finished:          finished, | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			baseCtxFn:       context.Background, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:   defaultIsFailureFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          defaultQueueConfig, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = tc.handler | ||||
| 		var ( | ||||
| @@ -695,21 +692,20 @@ func TestProcessorWithStrictPriority(t *testing.T) { | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:            testLogger, | ||||
| 			broker:            rdbClient, | ||||
| 			baseCtxFn:         context.Background, | ||||
| 			taskCheckInterval: defaultTaskCheckInterval, | ||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:     defaultIsFailureFunc, | ||||
| 			syncCh:            syncCh, | ||||
| 			cancelations:      base.NewCancelations(), | ||||
| 			concurrency:       1, // Set concurrency to 1 to make sure tasks are processed one at a time. | ||||
| 			queues:            queueCfg, | ||||
| 			strictPriority:    true, | ||||
| 			errHandler:        nil, | ||||
| 			shutdownTimeout:   defaultShutdownTimeout, | ||||
| 			starting:          starting, | ||||
| 			finished:          finished, | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			baseCtxFn:       context.Background, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:   defaultIsFailureFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     1, // Set concurrency to 1 to make sure tasks are processed one at a time. | ||||
| 			queues:          queueCfg, | ||||
| 			strictPriority:  true, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = HandlerFunc(handler) | ||||
|  | ||||
| @@ -925,45 +921,3 @@ func TestProcessorComputeDeadline(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestReturnPanicError(t *testing.T) { | ||||
|  | ||||
| 	task := NewTask("gen_thumbnail", h.JSON(map[string]interface{}{"src": "some/img/path"})) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name         string | ||||
| 		handler      HandlerFunc | ||||
| 		IsPanicError bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "should return panic error when occurred panic recovery", | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				panic("something went terribly wrong") | ||||
| 			}, | ||||
| 			IsPanicError: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "should return normal error when don't occur panic recovery", | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				return fmt.Errorf("something went terribly wrong") | ||||
| 			}, | ||||
| 			IsPanicError: false, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			p := processor{ | ||||
| 				logger:  log.NewLogger(nil), | ||||
| 				handler: tc.handler, | ||||
| 			} | ||||
| 			got := p.perform(context.Background(), task) | ||||
| 			if tc.IsPanicError != IsPanicError(got) { | ||||
| 				t.Errorf("%s: got=%t, want=%t", tc.name, IsPanicError(got), tc.IsPanicError) | ||||
| 			} | ||||
| 			if tc.IsPanicError && !strings.HasPrefix(got.Error(), "panic error cause by:") { | ||||
| 				t.Error("wrong text msg for panic error") | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										26
									
								
								scheduler.go
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								scheduler.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/robfig/cron/v3" | ||||
| ) | ||||
|  | ||||
| @@ -43,27 +43,15 @@ type Scheduler struct { | ||||
| 	// to avoid using cron.EntryID as the public API of | ||||
| 	// the Scheduler. | ||||
| 	idmap map[string]cron.EntryID | ||||
| 	// When a Scheduler has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
| } | ||||
|  | ||||
| // NewScheduler returns a new Scheduler instance given the redis connection option. | ||||
| // The parameter opts is optional, defaults will be used if opts is set to nil | ||||
| func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	scheduler := NewSchedulerFromRedisClient(redisClient, opts) | ||||
| 	scheduler.sharedConnection = false | ||||
| 	return scheduler | ||||
| } | ||||
|  | ||||
| // NewSchedulerFromRedisClient returns a new instance of Scheduler given a redis.UniversalClient | ||||
| // The parameter opts is optional, defaults will be used if opts is set to nil. | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewSchedulerFromRedisClient(c redis.UniversalClient, opts *SchedulerOpts) *Scheduler { | ||||
| 	if opts == nil { | ||||
| 		opts = &SchedulerOpts{} | ||||
| 	} | ||||
| @@ -84,7 +72,7 @@ func NewSchedulerFromRedisClient(c redis.UniversalClient, opts *SchedulerOpts) * | ||||
| 		id:              generateSchedulerID(), | ||||
| 		state:           &serverState{value: srvStateNew}, | ||||
| 		logger:          logger, | ||||
| 		client:          NewClientFromRedisClient(c), | ||||
| 		client:          NewClient(r), | ||||
| 		rdb:             rdb.NewRDB(c), | ||||
| 		cron:            cron.New(cron.WithLocation(loc)), | ||||
| 		location:        loc, | ||||
| @@ -273,12 +261,8 @@ func (s *Scheduler) Shutdown() { | ||||
| 	s.wg.Wait() | ||||
|  | ||||
| 	s.clearHistory() | ||||
| 	if err := s.client.Close(); err != nil { | ||||
| 		s.logger.Errorf("Failed to close redis client connection: %v", err) | ||||
| 	} | ||||
| 	if !s.sharedConnection { | ||||
| 		s.rdb.Close() | ||||
| 	} | ||||
| 	s.client.Close() | ||||
| 	s.rdb.Close() | ||||
| 	s.logger.Info("Scheduler stopped") | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,6 @@ | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @@ -59,7 +58,6 @@ func TestSchedulerRegister(t *testing.T) { | ||||
|  | ||||
| 	r := setup(t) | ||||
|  | ||||
| 	// Tests for new redis connection. | ||||
| 	for _, tc := range tests { | ||||
| 		scheduler := NewScheduler(getRedisConnOpt(t), nil) | ||||
| 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | ||||
| @@ -77,28 +75,6 @@ func TestSchedulerRegister(t *testing.T) { | ||||
| 			t.Errorf("mismatch found in queue %q: (-want,+got)\n%s", tc.queue, diff) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	r = setup(t) | ||||
|  | ||||
| 	// Tests for existing redis connection. | ||||
| 	for _, tc := range tests { | ||||
| 		redisClient := getRedisConnOpt(t).MakeRedisClient().(redis.UniversalClient) | ||||
| 		scheduler := NewSchedulerFromRedisClient(redisClient, nil) | ||||
| 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		if err := scheduler.Start(); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		time.Sleep(tc.wait) | ||||
| 		scheduler.Shutdown() | ||||
|  | ||||
| 		got := testutil.GetPendingMessages(t, r, tc.queue) | ||||
| 		if diff := cmp.Diff(tc.want, got, testutil.IgnoreIDOpt); diff != "" { | ||||
| 			t.Errorf("mismatch found in queue %q: (-want,+got)\n%s", tc.queue, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestSchedulerWhenRedisDown(t *testing.T) { | ||||
|   | ||||
							
								
								
									
										139
									
								
								server.go
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								server.go
									
									
									
									
									
								
							| @@ -15,10 +15,10 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // Server is responsible for task processing and task lifecycle management. | ||||
| @@ -37,9 +37,6 @@ type Server struct { | ||||
| 	logger *log.Logger | ||||
|  | ||||
| 	broker base.Broker | ||||
| 	// When a Server has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
|  | ||||
| 	state *serverState | ||||
|  | ||||
| @@ -107,15 +104,6 @@ type Config struct { | ||||
| 	// If this is defined, then it MUST return a non-nil context | ||||
| 	BaseContext func() context.Context | ||||
|  | ||||
| 	// TaskCheckInterval specifies the interval between checks for new tasks to process when all queues are empty. | ||||
| 	// | ||||
| 	// If unset, zero or a negative value, the interval is set to 1 second. | ||||
| 	// | ||||
| 	// Note: Setting this value too low may add significant load to redis. | ||||
| 	// | ||||
| 	// By default, TaskCheckInterval is set to 1 seconds. | ||||
| 	TaskCheckInterval time.Duration | ||||
|  | ||||
| 	// Function to calculate retry delay for a failed task. | ||||
| 	// | ||||
| 	// By default, it uses exponential backoff algorithm to calculate the delay. | ||||
| @@ -174,16 +162,6 @@ type Config struct { | ||||
| 	//     }) | ||||
| 	// | ||||
| 	//     ErrorHandler: asynq.ErrorHandlerFunc(reportError) | ||||
|  | ||||
| 	//    we can also handle panic error like: | ||||
| 	//     func reportError(ctx context, task *asynq.Task, err error) { | ||||
| 	//         if asynq.IsPanic(err) { | ||||
| 	//	          errorReportingService.Notify(err) | ||||
| 	// 	       } | ||||
| 	//     }) | ||||
| 	// | ||||
| 	//     ErrorHandler: asynq.ErrorHandlerFunc(reportError) | ||||
|  | ||||
| 	ErrorHandler ErrorHandler | ||||
|  | ||||
| 	// Logger specifies the logger used by the server instance. | ||||
| @@ -242,17 +220,6 @@ type Config struct { | ||||
| 	// | ||||
| 	// If unset or nil, the group aggregation feature will be disabled on the server. | ||||
| 	GroupAggregator GroupAggregator | ||||
|  | ||||
| 	// JanitorInterval specifies the average interval of janitor checks for expired completed tasks. | ||||
| 	// | ||||
| 	// If unset or zero, default interval of 8 seconds is used. | ||||
| 	JanitorInterval time.Duration | ||||
|  | ||||
| 	// JanitorBatchSize specifies the number of expired completed tasks to be deleted in one run. | ||||
| 	// | ||||
| 	// If unset or zero, default batch size of 100 is used. | ||||
| 	// Make sure to not put a big number as the batch size to prevent a long-running script. | ||||
| 	JanitorBatchSize int | ||||
| } | ||||
|  | ||||
| // GroupAggregator aggregates a group of tasks into one before the tasks are passed to the Handler. | ||||
| @@ -413,8 +380,6 @@ var defaultQueueConfig = map[string]int{ | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	defaultTaskCheckInterval = 1 * time.Second | ||||
|  | ||||
| 	defaultShutdownTimeout = 8 * time.Second | ||||
|  | ||||
| 	defaultHealthCheckInterval = 15 * time.Second | ||||
| @@ -422,28 +387,15 @@ const ( | ||||
| 	defaultDelayedTaskCheckInterval = 5 * time.Second | ||||
|  | ||||
| 	defaultGroupGracePeriod = 1 * time.Minute | ||||
|  | ||||
| 	defaultJanitorInterval = 8 * time.Second | ||||
|  | ||||
| 	defaultJanitorBatchSize = 100 | ||||
| ) | ||||
|  | ||||
| // NewServer returns a new Server given a redis connection option | ||||
| // and server configuration. | ||||
| func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	server := NewServerFromRedisClient(redisClient, cfg) | ||||
| 	server.sharedConnection = false | ||||
| 	return server | ||||
| } | ||||
|  | ||||
| // NewServerFromRedisClient returns a new instance of Server given a redis.UniversalClient | ||||
| // and server configuration | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 	baseCtxFn := cfg.BaseContext | ||||
| 	if baseCtxFn == nil { | ||||
| 		baseCtxFn = context.Background | ||||
| @@ -452,12 +404,6 @@ func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 	if n < 1 { | ||||
| 		n = runtime.NumCPU() | ||||
| 	} | ||||
|  | ||||
| 	taskCheckInterval := cfg.TaskCheckInterval | ||||
| 	if taskCheckInterval <= 0 { | ||||
| 		taskCheckInterval = defaultTaskCheckInterval | ||||
| 	} | ||||
|  | ||||
| 	delayFunc := cfg.RetryDelayFunc | ||||
| 	if delayFunc == nil { | ||||
| 		delayFunc = DefaultRetryDelayFunc | ||||
| @@ -544,21 +490,20 @@ func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 		cancelations: cancels, | ||||
| 	}) | ||||
| 	processor := newProcessor(processorParams{ | ||||
| 		logger:            logger, | ||||
| 		broker:            rdb, | ||||
| 		retryDelayFunc:    delayFunc, | ||||
| 		taskCheckInterval: taskCheckInterval, | ||||
| 		baseCtxFn:         baseCtxFn, | ||||
| 		isFailureFunc:     isFailureFunc, | ||||
| 		syncCh:            syncCh, | ||||
| 		cancelations:      cancels, | ||||
| 		concurrency:       n, | ||||
| 		queues:            queues, | ||||
| 		strictPriority:    cfg.StrictPriority, | ||||
| 		errHandler:        cfg.ErrorHandler, | ||||
| 		shutdownTimeout:   shutdownTimeout, | ||||
| 		starting:          starting, | ||||
| 		finished:          finished, | ||||
| 		logger:          logger, | ||||
| 		broker:          rdb, | ||||
| 		retryDelayFunc:  delayFunc, | ||||
| 		baseCtxFn:       baseCtxFn, | ||||
| 		isFailureFunc:   isFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    cancels, | ||||
| 		concurrency:     n, | ||||
| 		queues:          queues, | ||||
| 		strictPriority:  cfg.StrictPriority, | ||||
| 		errHandler:      cfg.ErrorHandler, | ||||
| 		shutdownTimeout: shutdownTimeout, | ||||
| 		starting:        starting, | ||||
| 		finished:        finished, | ||||
| 	}) | ||||
| 	recoverer := newRecoverer(recovererParams{ | ||||
| 		logger:         logger, | ||||
| @@ -574,26 +519,11 @@ func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 		interval:        healthcheckInterval, | ||||
| 		healthcheckFunc: cfg.HealthCheckFunc, | ||||
| 	}) | ||||
|  | ||||
| 	janitorInterval := cfg.JanitorInterval | ||||
| 	if janitorInterval == 0 { | ||||
| 		janitorInterval = defaultJanitorInterval | ||||
| 	} | ||||
|  | ||||
| 	janitorBatchSize := cfg.JanitorBatchSize | ||||
| 	if janitorBatchSize == 0 { | ||||
| 		janitorBatchSize = defaultJanitorBatchSize | ||||
| 	} | ||||
| 	if janitorBatchSize > defaultJanitorBatchSize { | ||||
| 		logger.Warnf("Janitor batch size of %d is greater than the recommended batch size of %d. "+ | ||||
| 			"This might cause a long-running script", janitorBatchSize, defaultJanitorBatchSize) | ||||
| 	} | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:    logger, | ||||
| 		broker:    rdb, | ||||
| 		queues:    qnames, | ||||
| 		interval:  janitorInterval, | ||||
| 		batchSize: janitorBatchSize, | ||||
| 		logger:   logger, | ||||
| 		broker:   rdb, | ||||
| 		queues:   qnames, | ||||
| 		interval: 8 * time.Second, | ||||
| 	}) | ||||
| 	aggregator := newAggregator(aggregatorParams{ | ||||
| 		logger:          logger, | ||||
| @@ -605,19 +535,18 @@ func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 		groupAggregator: cfg.GroupAggregator, | ||||
| 	}) | ||||
| 	return &Server{ | ||||
| 		logger:           logger, | ||||
| 		broker:           rdb, | ||||
| 		sharedConnection: true, | ||||
| 		state:            srvState, | ||||
| 		forwarder:        forwarder, | ||||
| 		processor:        processor, | ||||
| 		syncer:           syncer, | ||||
| 		heartbeater:      heartbeater, | ||||
| 		subscriber:       subscriber, | ||||
| 		recoverer:        recoverer, | ||||
| 		healthchecker:    healthchecker, | ||||
| 		janitor:          janitor, | ||||
| 		aggregator:       aggregator, | ||||
| 		logger:        logger, | ||||
| 		broker:        rdb, | ||||
| 		state:         srvState, | ||||
| 		forwarder:     forwarder, | ||||
| 		processor:     processor, | ||||
| 		syncer:        syncer, | ||||
| 		heartbeater:   heartbeater, | ||||
| 		subscriber:    subscriber, | ||||
| 		recoverer:     recoverer, | ||||
| 		healthchecker: healthchecker, | ||||
| 		janitor:       janitor, | ||||
| 		aggregator:    aggregator, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -745,9 +674,7 @@ func (srv *Server) Shutdown() { | ||||
| 	srv.heartbeater.shutdown() | ||||
| 	srv.wg.Wait() | ||||
|  | ||||
| 	if !srv.sharedConnection { | ||||
| 		srv.broker.Close() | ||||
| 	} | ||||
| 	srv.broker.Close() | ||||
| 	srv.logger.Info("Exiting") | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -14,12 +14,22 @@ import ( | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/hibiken/asynq/internal/testbroker" | ||||
| 	"github.com/hibiken/asynq/internal/testutil" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"go.uber.org/goleak" | ||||
| ) | ||||
|  | ||||
| func testServer(t *testing.T, c *Client, srv *Server) { | ||||
| func TestServer(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	c := NewClient(redisConnOpt) | ||||
| 	defer c.Close() | ||||
| 	srv := NewServer(redisConnOpt, Config{ | ||||
| 		Concurrency: 10, | ||||
| 		LogLevel:    testLogLevel, | ||||
| 	}) | ||||
|  | ||||
| 	// no-op handler | ||||
| 	h := func(ctx context.Context, task *Task) error { | ||||
| 		return nil | ||||
| @@ -43,43 +53,6 @@ func testServer(t *testing.T, c *Client, srv *Server) { | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestServer(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	c := NewClient(redisConnOpt) | ||||
| 	defer c.Close() | ||||
| 	srv := NewServer(redisConnOpt, Config{ | ||||
| 		Concurrency: 10, | ||||
| 		LogLevel:    testLogLevel, | ||||
| 	}) | ||||
|  | ||||
| 	testServer(t, c, srv) | ||||
| } | ||||
|  | ||||
| func TestServerFromRedisClient(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	redisClient := redisConnOpt.MakeRedisClient().(redis.UniversalClient) | ||||
| 	c := NewClientFromRedisClient(redisClient) | ||||
| 	srv := NewServerFromRedisClient(redisClient, Config{ | ||||
| 		Concurrency: 10, | ||||
| 		LogLevel:    testLogLevel, | ||||
| 	}) | ||||
|  | ||||
| 	testServer(t, c, srv) | ||||
|  | ||||
| 	err := c.Close() | ||||
| 	if err == nil { | ||||
| 		t.Error("client.Close() should have failed because of a shared client but it didn't") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestServerRun(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| //go:build linux || dragonfly || freebsd || netbsd || openbsd || darwin | ||||
| // +build linux bsd darwin | ||||
|  | ||||
| package asynq | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| //go:build windows | ||||
| // +build windows | ||||
|  | ||||
| package asynq | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ Asynq CLI is a command line tool to monitor the queues and tasks managed by `asy | ||||
|  | ||||
| In order to use the tool, compile it using the following command: | ||||
|  | ||||
|     go install github.com/hibiken/asynq/tools/asynq@latest | ||||
|     go install github.com/hibiken/asynq/tools/asynq | ||||
|  | ||||
| This will create the asynq executable under your `$GOPATH/bin` directory. | ||||
|  | ||||
| @@ -25,7 +25,7 @@ To view details on any command, use `asynq help <command> <subcommand>`. | ||||
| - `asynq dash` | ||||
| - `asynq stats` | ||||
| - `asynq queue [ls inspect history rm pause unpause]` | ||||
| - `asynq task [ls cancel delete archive run deleteall archiveall runall]` | ||||
| - `asynq task [ls cancel delete archive run delete-all archive-all run-all]` | ||||
| - `asynq server [ls]` | ||||
|  | ||||
| ### Global flags | ||||
|   | ||||
| @@ -369,7 +369,7 @@ func createRDB() *rdb.RDB { | ||||
| 	return rdb.NewRDB(c) | ||||
| } | ||||
|  | ||||
| // createInspector creates a Inspector instance using flag values and returns it. | ||||
| // createRDB creates a Inspector instance using flag values and returns it. | ||||
| func createInspector() *asynq.Inspector { | ||||
| 	return asynq.NewInspector(getRedisConnOpt()) | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| module github.com/hibiken/asynq/tools | ||||
|  | ||||
| go 1.20 | ||||
| go 1.18 | ||||
|  | ||||
| require ( | ||||
| 	github.com/MakeNowJust/heredoc/v2 v2.0.1 | ||||
| @@ -11,7 +11,7 @@ require ( | ||||
| 	github.com/hibiken/asynq/x v0.0.0-20220131170841-349f4c50fb1d | ||||
| 	github.com/mattn/go-runewidth v0.0.13 | ||||
| 	github.com/mitchellh/go-homedir v1.1.0 | ||||
| 	github.com/prometheus/client_golang v1.11.1 | ||||
| 	github.com/prometheus/client_golang v1.11.0 | ||||
| 	github.com/redis/go-redis/v9 v9.0.5 | ||||
| 	github.com/spf13/cobra v1.1.1 | ||||
| 	github.com/spf13/pflag v1.0.5 | ||||
| @@ -47,7 +47,7 @@ require ( | ||||
| 	github.com/subosito/gotenv v1.2.0 // indirect | ||||
| 	golang.org/x/sys v0.10.0 // indirect | ||||
| 	golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf // indirect | ||||
| 	golang.org/x/text v0.3.8 // indirect | ||||
| 	golang.org/x/text v0.3.7 // indirect | ||||
| 	golang.org/x/time v0.3.0 // indirect | ||||
| 	google.golang.org/protobuf v1.31.0 // indirect | ||||
| 	gopkg.in/ini.v1 v1.51.0 // indirect | ||||
|   | ||||
| @@ -235,9 +235,8 @@ github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXP | ||||
| github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso= | ||||
| github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.11.0 h1:HNkLOAEQMIDv/K+04rukrLx6ch7msSRwf3/SASFAGtQ= | ||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_golang v1.11.1 h1:+4eQaD7vAZ6DsfsxB15hbE0odUjGI5ARs9yskGu1v4s= | ||||
| github.com/prometheus/client_golang v1.11.1/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| @@ -424,9 +423,8 @@ golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3 | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.3.8 h1:nAL+RVCQ9uMn3vJZbV+MRnydTJFPf8qqY42YiA6MrqY= | ||||
| golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||
| golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= | ||||
|   | ||||
							
								
								
									
										26
									
								
								x/go.mod
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								x/go.mod
									
									
									
									
									
								
							| @@ -1,26 +1,10 @@ | ||||
| module github.com/hibiken/asynq/x | ||||
|  | ||||
| go 1.20 | ||||
| go 1.16 | ||||
|  | ||||
| require ( | ||||
| 	github.com/google/uuid v1.4.0 | ||||
| 	github.com/hibiken/asynq v0.24.1 | ||||
| 	github.com/prometheus/client_golang v1.11.1 | ||||
| 	github.com/redis/go-redis/v9 v9.3.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/beorn7/perks v1.0.1 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/golang/protobuf v1.5.3 // indirect | ||||
| 	github.com/matttproud/golang_protobuf_extensions v1.0.1 // indirect | ||||
| 	github.com/prometheus/client_model v0.2.0 // indirect | ||||
| 	github.com/prometheus/common v0.26.0 // indirect | ||||
| 	github.com/prometheus/procfs v0.6.0 // indirect | ||||
| 	github.com/robfig/cron/v3 v3.0.1 // indirect | ||||
| 	github.com/spf13/cast v1.6.0 // indirect | ||||
| 	golang.org/x/sys v0.15.0 // indirect | ||||
| 	golang.org/x/time v0.5.0 // indirect | ||||
| 	google.golang.org/protobuf v1.31.0 // indirect | ||||
| 	github.com/go-redis/redis/v8 v8.11.4 | ||||
| 	github.com/google/uuid v1.3.0 | ||||
| 	github.com/hibiken/asynq v0.21.0 | ||||
| 	github.com/prometheus/client_golang v1.11.0 | ||||
| ) | ||||
|   | ||||
							
								
								
									
										147
									
								
								x/go.sum
									
									
									
									
									
								
							
							
						
						
									
										147
									
								
								x/go.sum
									
									
									
									
									
								
							| @@ -1,4 +1,6 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| @@ -8,26 +10,35 @@ github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24 | ||||
| github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= | ||||
| github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= | ||||
| github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= | ||||
| github.com/bsm/ginkgo/v2 v2.7.0/go.mod h1:AiKlXPm7ItEHNc/2+OkrNG4E0ITzojb9/xWzvQ9XZ9w= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/gomega v1.26.0/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | ||||
| github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
| github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= | ||||
| github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= | ||||
| github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | ||||
| github.com/go-redis/redis/v8 v8.11.2/go.mod h1:DLomh7y2e3ggQXQLd1YgmvIfecPJoFl7WU5SOQ/r06M= | ||||
| github.com/go-redis/redis/v8 v8.11.4 h1:kHoYkfZP6+pe04aFTnhDH6GDROa5yJdHJVNxV3F46Tg= | ||||
| github.com/go-redis/redis/v8 v8.11.4/go.mod h1:2Z2wHZXdQpCDXEGzqMockDpNyYvi2l4Pxt6RJr792+w= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| @@ -36,25 +47,28 @@ github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:x | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= | ||||
| github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.6 h1:BKbKCqvP6I+rmFHt06ZmyQtvB8xAkWdhFyr0ZUNZcxQ= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= | ||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.4.0 h1:MtMxsa51/r9yyhkyLsVeVt0B+BGQZzpQiTQ4eHZ8bc4= | ||||
| github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hibiken/asynq v0.24.1 h1:+5iIEAyA9K/lcSPvx3qoPtsKJeKI5u9aOIvUmSsazEw= | ||||
| github.com/hibiken/asynq v0.24.1/go.mod h1:u5qVeSbrnfT+vtG5Mq8ZPzQu/BmCKMHvTGb91uy9Tts= | ||||
| github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= | ||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hibiken/asynq v0.21.0 h1:uH9XogJhjq/S39E0/DEPWLZQ6hHJ73UiblZTe4RzHwA= | ||||
| github.com/hibiken/asynq v0.21.0/go.mod h1:tyc63ojaW8SJ5SBm8mvI4DDONsguP5HE85EEl4Qr5Ig= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= | ||||
| github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||
| github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| @@ -65,10 +79,8 @@ github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxv | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| @@ -77,17 +89,32 @@ github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lN | ||||
| github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= | ||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.15.0/go.mod h1:hF8qUzuuC8DJGygJH3726JnCZX4MYbRB8yFfISqnKUg= | ||||
| github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= | ||||
| github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= | ||||
| github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= | ||||
| github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= | ||||
| github.com/onsi/gomega v1.10.5/go.mod h1:gza4q3jKQJijlu05nKWRCW/GavJumGt8aNRxWg7mt48= | ||||
| github.com/onsi/gomega v1.16.0 h1:6gjqkI8iiRHMvdccRJM8rVKjCWk6ZIm6FTm3ddIe4/c= | ||||
| github.com/onsi/gomega v1.16.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= | ||||
| github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= | ||||
| github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.11.1 h1:+4eQaD7vAZ6DsfsxB15hbE0odUjGI5ARs9yskGu1v4s= | ||||
| github.com/prometheus/client_golang v1.11.1/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_golang v1.11.0 h1:HNkLOAEQMIDv/K+04rukrLx6ch7msSRwf3/SASFAGtQ= | ||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= | ||||
| github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| @@ -99,99 +126,133 @@ github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsT | ||||
| github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= | ||||
| github.com/prometheus/procfs v0.6.0 h1:mxy4L2jP6qMonqmq+aTtOx1ifVWUgG/TAmntgbh3xv4= | ||||
| github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||
| github.com/redis/go-redis/v9 v9.0.3/go.mod h1:WqMKv5vnQbRuZstUwxQI195wHy+t4PuXDOjzMvcuQHk= | ||||
| github.com/redis/go-redis/v9 v9.3.0 h1:RiVDjmig62jIWp7Kk4XVLs0hzV6pI3PyTnnL0cnn0u0= | ||||
| github.com/redis/go-redis/v9 v9.3.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M= | ||||
| github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= | ||||
| github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | ||||
| github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||||
| github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||
| github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= | ||||
| github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= | ||||
| github.com/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= | ||||
| github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | ||||
| github.com/spf13/cast v1.6.0 h1:GEiTHELF+vaR5dhz3VqZfFSzZjYbgeKDpBxQVS4GYJ0= | ||||
| github.com/spf13/cast v1.6.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||
| go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= | ||||
| go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| go.uber.org/goleak v0.10.0 h1:G3eWbSNIskeRqtsN/1uI5B+eP73y3JUuBsv9AZjehb4= | ||||
| go.uber.org/goleak v0.10.0/go.mod h1:VCZuO8V8mFPlL0F5J5GK1rtHV3DrFcQ1R8ryq7FK0aI= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781 h1:DzZ89McO9/gWPsQXS/FVKAlG02ZjaQ6AlZRBimEYOd0= | ||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40 h1:JWgyZ1qgdTaF3N3oxC+MdTV7qvEEgHo3otj+HB5CM7Q= | ||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.15.0 h1:h48lPFYpsTvQJZF4EKyI4aLHaev3CxivZmv7yZig9pc= | ||||
| golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk= | ||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= | ||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8= | ||||
| google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
|   | ||||
| @@ -7,9 +7,9 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // NewSemaphore creates a counting Semaphore for the given scope with the given number of tokens. | ||||
|   | ||||
| @@ -8,11 +8,11 @@ import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| @@ -91,7 +91,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | ||||
| 			maxConcurrency: 3, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-1", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -104,7 +104,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | ||||
| 			maxConcurrency: 3, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString(), uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-2", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -211,7 +211,7 @@ func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | ||||
|  | ||||
| 	// adding a set member to mimic the case where token is acquired but the goroutine crashed, | ||||
| 	// in which case, the token will not be explicitly removed and should be present already | ||||
| 	rc.ZAdd(context.Background(), semaphoreKey("stale-token"), redis.Z{ | ||||
| 	rc.ZAdd(context.Background(), semaphoreKey("stale-token"), &redis.Z{ | ||||
| 		Score:  float64(time.Now().Add(-10 * time.Second).Unix()), | ||||
| 		Member: taskID, | ||||
| 	}) | ||||
| @@ -219,7 +219,7 @@ func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | ||||
| 	sema := NewSemaphore(opt, "stale-token", 1) | ||||
| 	defer sema.Close() | ||||
|  | ||||
| 	ctx, cancel := asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 	ctx, cancel := asynqcontext.New(&base.TaskMessage{ | ||||
| 		ID:    taskID, | ||||
| 		Queue: "task-1", | ||||
| 	}, time.Now().Add(time.Second)) | ||||
| @@ -248,7 +248,7 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 			name:    "task-5", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-3", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -259,7 +259,7 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 			name:    "task-6", | ||||
| 			taskIDs: []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -277,9 +277,9 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []redis.Z | ||||
| 			var members []*redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, redis.Z{ | ||||
| 				members = append(members, &redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
| @@ -337,7 +337,7 @@ func TestNewSemaphore_Release_Error(t *testing.T) { | ||||
| 			name:    "task-8", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    testID, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -356,9 +356,9 @@ func TestNewSemaphore_Release_Error(t *testing.T) { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []redis.Z | ||||
| 			var members []*redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, redis.Z{ | ||||
| 				members = append(members, &redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user