mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-21 21:46:12 +08:00 
			
		
		
		
	Compare commits
	
		
			35 Commits
		
	
	
		
			revert-491
			...
			develop
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 461d922616 | ||
|  | 5daa3c52ed | ||
|  | d04888e748 | ||
|  | 174008843d | ||
|  | 2b632b93d5 | ||
|  | b35b559d40 | ||
|  | 8df0bfa583 | ||
|  | b25d10b61d | ||
|  | 38f7499b71 | ||
|  | 0a73fc6201 | ||
|  | 1a11a33b4f | ||
|  | f0888df813 | ||
|  | c2dd648a51 | ||
|  | a3cca853a0 | ||
|  | 83df622a92 | ||
|  | fdbf54eb04 | ||
|  | 16ec43cbca | ||
|  | 1e0bf88bf3 | ||
|  | d0041c55a3 | ||
|  | 7ef0511f35 | ||
|  | 1ec90810db | ||
|  | 90188a093d | ||
|  | e05f0b7196 | ||
|  | c1096a0fae | ||
|  | 9e548fc097 | ||
|  | 6a7bf2ceff | ||
|  | e7fa0ae865 | ||
|  | fc4b6713f6 | ||
|  | 6b98c0bbae | ||
|  | ed1ab8ee55 | ||
|  | e18c0381ad | ||
|  | 8b422c237c | ||
|  | e6f74c1c2b | ||
|  | 6edba6994e | ||
|  | 571f0d2613 | 
							
								
								
									
										21
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | 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 |     runs-on: ubuntu-latest | ||||||
|     services: |     services: | ||||||
|       redis: |       redis: | ||||||
|         image: redis |         image: redis:7 | ||||||
|         ports: |         ports: | ||||||
|           - 6379:6379 |           - 6379:6379 | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v2 |         uses: actions/checkout@v4 | ||||||
|       - name: Set up Go |       - name: Set up Go | ||||||
|         uses: actions/setup-go@v2 |         uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version: 1.16.x |           go-version: 1.21.x | ||||||
|       - name: Benchmark |       - name: Benchmark | ||||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a new.txt |         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a new.txt | ||||||
|       - name: Upload Benchmark |       - name: Upload Benchmark | ||||||
|         uses: actions/upload-artifact@v2 |         uses: actions/upload-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           name: bench-incoming |           name: bench-incoming | ||||||
|           path: new.txt |           path: new.txt | ||||||
| @@ -33,22 +33,22 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     services: |     services: | ||||||
|       redis: |       redis: | ||||||
|         image: redis |         image: redis:7 | ||||||
|         ports: |         ports: | ||||||
|           - 6379:6379 |           - 6379:6379 | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v2 |         uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           ref: master |           ref: master | ||||||
|       - name: Set up Go |       - name: Set up Go | ||||||
|         uses: actions/setup-go@v2 |         uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version: 1.15.x |           go-version: 1.21.x | ||||||
|       - name: Benchmark |       - name: Benchmark | ||||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a old.txt |         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a old.txt | ||||||
|       - name: Upload Benchmark |       - name: Upload Benchmark | ||||||
|         uses: actions/upload-artifact@v2 |         uses: actions/upload-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           name: bench-current |           name: bench-current | ||||||
|           path: old.txt |           path: old.txt | ||||||
| @@ -58,25 +58,25 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v2 |         uses: actions/checkout@v4 | ||||||
|       - name: Set up Go |       - name: Set up Go | ||||||
|         uses: actions/setup-go@v2 |         uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version: 1.15.x |           go-version: 1.21.x | ||||||
|       - name: Install benchstat |       - name: Install benchstat | ||||||
|         run: go get -u golang.org/x/perf/cmd/benchstat |         run: go get -u golang.org/x/perf/cmd/benchstat | ||||||
|       - name: Download Incoming |       - name: Download Incoming | ||||||
|         uses: actions/download-artifact@v2 |         uses: actions/download-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           name: bench-incoming |           name: bench-incoming | ||||||
|       - name: Download Current |       - name: Download Current | ||||||
|         uses: actions/download-artifact@v2 |         uses: actions/download-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           name: bench-current |           name: bench-current | ||||||
|       - name: Benchstat Results |       - name: Benchstat Results | ||||||
|         run: benchstat old.txt new.txt | tee -a benchstat.txt |         run: benchstat old.txt new.txt | tee -a benchstat.txt | ||||||
|       - name: Upload benchstat results |       - name: Upload benchstat results | ||||||
|         uses: actions/upload-artifact@v2 |         uses: actions/upload-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           name: benchstat |           name: benchstat | ||||||
|           path: benchstat.txt |           path: benchstat.txt | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,20 +7,21 @@ jobs: | |||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest] |         os: [ubuntu-latest] | ||||||
|         go-version: [1.14.x, 1.15.x, 1.16.x, 1.17.x, 1.18.x] |         go-version: [1.20.x, 1.21.x] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     services: |     services: | ||||||
|       redis: |       redis: | ||||||
|         image: redis |         image: redis:7 | ||||||
|         ports: |         ports: | ||||||
|           - 6379:6379 |           - 6379:6379 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - name: Set up Go |       - name: Set up Go | ||||||
|         uses: actions/setup-go@v2 |         uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version: ${{ matrix.go-version }} |           go-version: ${{ matrix.go-version }} | ||||||
|  |           cache: false | ||||||
|  |  | ||||||
|       - name: Build core module |       - name: Build core module | ||||||
|         run: go build -v ./... |         run: go build -v ./... | ||||||
| @@ -44,20 +45,21 @@ jobs: | |||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest] |         os: [ubuntu-latest] | ||||||
|         go-version: [1.18.x] |         go-version: [1.20.x, 1.21.x] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     services: |     services: | ||||||
|       redis: |       redis: | ||||||
|         image: redis |         image: redis:7 | ||||||
|         ports: |         ports: | ||||||
|           - 6379:6379 |           - 6379:6379 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|       - name: Set up Go |       - name: Set up Go | ||||||
|         uses: actions/setup-go@v2 |         uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version: ${{ matrix.go-version }} |           go-version: ${{ matrix.go-version }} | ||||||
|  |           cache: false | ||||||
|  |  | ||||||
|       - name: Build tools module |       - name: Build tools module | ||||||
|         run: cd tools && go build -v ./... && cd .. |         run: cd tools && go build -v ./... && cd .. | ||||||
|   | |||||||
| @@ -7,6 +7,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | |||||||
|  |  | ||||||
| ## [Unreleased] | ## [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 | ## [0.24.1] - 2023-05-01 | ||||||
|  |  | ||||||
| ### Changed | ### Changed | ||||||
|   | |||||||
| @@ -54,7 +54,7 @@ If you are using this package in production, **please consider sponsoring the pr | |||||||
|  |  | ||||||
| ## Quickstart | ## Quickstart | ||||||
|  |  | ||||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). Version `1.14` or higher is required. | Make sure you have Go installed ([download](https://golang.org/dl/)). Latest two Go versions are supported (See https://go.dev/dl). | ||||||
|  |  | ||||||
| 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: | 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: | To install the CLI tool, run the following command: | ||||||
|  |  | ||||||
| ```sh | ```sh | ||||||
| go install github.com/hibiken/asynq/tools/asynq | go install github.com/hibiken/asynq/tools/asynq@latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Here's an example of running the `asynq dash` command: | Here's an example of running the `asynq dash` command: | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								client.go
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								client.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/redis/go-redis/v9" |  | ||||||
| 	"github.com/google/uuid" | 	"github.com/google/uuid" | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	"github.com/hibiken/asynq/internal/errors" | 	"github.com/hibiken/asynq/internal/errors" | ||||||
| 	"github.com/hibiken/asynq/internal/rdb" | 	"github.com/hibiken/asynq/internal/rdb" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // A Client is responsible for scheduling tasks. | // A Client is responsible for scheduling tasks. | ||||||
| @@ -25,15 +25,26 @@ import ( | |||||||
| // Clients are safe for concurrent use by multiple goroutines. | // Clients are safe for concurrent use by multiple goroutines. | ||||||
| type Client struct { | type Client struct { | ||||||
| 	broker base.Broker | 	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. | // NewClient returns a new Client instance given a redis connection option. | ||||||
| func NewClient(r RedisConnOpt) *Client { | func NewClient(r RedisConnOpt) *Client { | ||||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||||
| 	if !ok { | 	if !ok { | ||||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||||
| 	} | 	} | ||||||
| 	return &Client{broker: rdb.NewRDB(c)} | 	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} | ||||||
| } | } | ||||||
|  |  | ||||||
| type OptionType int | type OptionType int | ||||||
| @@ -307,6 +318,9 @@ var ( | |||||||
|  |  | ||||||
| // Close closes the connection with redis. | // Close closes the connection with redis. | ||||||
| func (c *Client) Close() error { | 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() | 	return c.broker.Close() | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ import ( | |||||||
| 	"github.com/google/go-cmp/cmp/cmpopts" | 	"github.com/google/go-cmp/cmp/cmpopts" | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	h "github.com/hibiken/asynq/internal/testutil" | 	h "github.com/hibiken/asynq/internal/testutil" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestClientEnqueueWithProcessAtOption(t *testing.T) { | func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||||
| @@ -143,11 +144,7 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestClientEnqueue(t *testing.T) { | func testClientEnqueue(t *testing.T, client *Client, r redis.UniversalClient) { | ||||||
| 	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"})) | 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||||
| 	now := time.Now() | 	now := time.Now() | ||||||
|  |  | ||||||
| @@ -478,6 +475,24 @@ func TestClientEnqueue(t *testing.T) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | 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) { | func TestClientEnqueueWithGroupOption(t *testing.T) { | ||||||
| 	r := setup(t) | 	r := setup(t) | ||||||
| 	client := NewClient(getRedisConnOpt(t)) | 	client := NewClient(getRedisConnOpt(t)) | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,17 +1,21 @@ | |||||||
| module github.com/hibiken/asynq | module github.com/hibiken/asynq | ||||||
|  |  | ||||||
| go 1.14 | go 1.20 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/golang/protobuf v1.5.2 | 	github.com/golang/protobuf v1.5.3 | ||||||
| 	github.com/google/go-cmp v0.5.6 | 	github.com/google/go-cmp v0.6.0 | ||||||
| 	github.com/google/uuid v1.2.0 | 	github.com/google/uuid v1.6.0 | ||||||
| 	github.com/redis/go-redis/v9 v9.0.3 | 	github.com/redis/go-redis/v9 v9.4.0 | ||||||
| 	github.com/robfig/cron/v3 v3.0.1 | 	github.com/robfig/cron/v3 v3.0.1 | ||||||
| 	github.com/spf13/cast v1.3.1 | 	github.com/spf13/cast v1.5.1 | ||||||
| 	go.uber.org/goleak v1.1.12 | 	go.uber.org/goleak v1.3.0 | ||||||
| 	golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e | 	golang.org/x/sys v0.16.0 | ||||||
| 	golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 | 	golang.org/x/time v0.3.0 | ||||||
| 	google.golang.org/protobuf v1.26.0 | 	google.golang.org/protobuf v1.31.0 | ||||||
| 	gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect | ) | ||||||
|  |  | ||||||
|  | require ( | ||||||
|  | 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||||
|  | 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,82 +1,39 @@ | |||||||
| github.com/bsm/ginkgo/v2 v2.7.0 h1:ItPMPH90RbmZJt5GtkcNvIRuGEdwlBItdNVoyzaNQao= | github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||||
| github.com/bsm/ginkgo/v2 v2.7.0/go.mod h1:AiKlXPm7ItEHNc/2+OkrNG4E0ITzojb9/xWzvQ9XZ9w= | github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||||
| 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 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | 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 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 h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | 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.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= | ||||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | github.com/golang/protobuf v1.5.3/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.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.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||||
| github.com/google/uuid v1.2.0 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | github.com/google/uuid v1.6.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.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||||
| 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/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | github.com/redis/go-redis/v9 v9.4.0 h1:Yzoz33UZw9I/mFhx4MNrB6Fk+XHO1VukNcCa1+lwyKk= | ||||||
| github.com/redis/go-redis/v9 v9.0.3 h1:+7mmR26M0IvyLxGZUHxu4GiBkJkVDid0Un+j4ScYu4k= | github.com/redis/go-redis/v9 v9.4.0/go.mod h1:hdY0cQFCN4fnSYT6TkisLufl/4W5UIXyv0b/CLO2V2M= | ||||||
| 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 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= | ||||||
| github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | ||||||
| github.com/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= | github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||||||
| github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | github.com/spf13/cast v1.5.1 h1:R+kOtfhWQE6TVQzY+4D7wJLBgkdVasCEFxSUBYBYIlA= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/spf13/cast v1.5.1/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48= | ||||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= | ||||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | golang.org/x/sys v0.16.0 h1:xWw16ngr6ZMtmxDyKyIgsE93KNKz5HKmMa3b8ALHidU= | ||||||
| go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= | golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||||
| go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= | ||||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | golang.org/x/time v0.3.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||||
| 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-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-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.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8= | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
| 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= |  | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								inspector.go
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								inspector.go
									
									
									
									
									
								
							| @@ -10,16 +10,19 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/redis/go-redis/v9" |  | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	"github.com/hibiken/asynq/internal/errors" | 	"github.com/hibiken/asynq/internal/errors" | ||||||
| 	"github.com/hibiken/asynq/internal/rdb" | 	"github.com/hibiken/asynq/internal/rdb" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Inspector is a client interface to inspect and mutate the state of | // Inspector is a client interface to inspect and mutate the state of | ||||||
| // queues and tasks. | // queues and tasks. | ||||||
| type Inspector struct { | type Inspector struct { | ||||||
| 	rdb *rdb.RDB | 	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. | // New returns a new instance of Inspector. | ||||||
| @@ -28,13 +31,25 @@ func NewInspector(r RedisConnOpt) *Inspector { | |||||||
| 	if !ok { | 	if !ok { | ||||||
| 		panic(fmt.Sprintf("inspeq: unsupported RedisConnOpt type %T", r)) | 		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{ | 	return &Inspector{ | ||||||
| 		rdb:              rdb.NewRDB(c), | 		rdb:              rdb.NewRDB(c), | ||||||
|  | 		sharedConnection: true, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // Close closes the connection with redis. | // Close closes the connection with redis. | ||||||
| func (i *Inspector) Close() error { | 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() | 	return i.rdb.Close() | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -22,11 +22,7 @@ import ( | |||||||
| 	"github.com/redis/go-redis/v9" | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestInspectorQueues(t *testing.T) { | func testInspectorQueues(t *testing.T, inspector *Inspector, r redis.UniversalClient) { | ||||||
| 	r := setup(t) |  | ||||||
| 	defer r.Close() |  | ||||||
| 	inspector := NewInspector(getRedisConnOpt(t)) |  | ||||||
|  |  | ||||||
| 	tests := []struct { | 	tests := []struct { | ||||||
| 		queues []string | 		queues []string | ||||||
| 	}{ | 	}{ | ||||||
| @@ -52,7 +48,21 @@ func TestInspectorQueues(t *testing.T) { | |||||||
| 			t.Errorf("Queues() = %v, want %v; (-want, +got)\n%s", got, tc.queues, diff) | 			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) { | func TestInspectorDeleteQueue(t *testing.T) { | ||||||
|   | |||||||
| @@ -737,7 +737,7 @@ type Broker interface { | |||||||
| 	ReclaimStaleAggregationSets(qname string) error | 	ReclaimStaleAggregationSets(qname string) error | ||||||
|  |  | ||||||
| 	// Task retention related method | 	// Task retention related method | ||||||
| 	DeleteExpiredCompletedTasks(qname string) error | 	DeleteExpiredCompletedTasks(qname string, batchSize int) error | ||||||
|  |  | ||||||
| 	// Lease related methods | 	// Lease related methods | ||||||
| 	ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*TaskMessage, error) | 	ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*TaskMessage, error) | ||||||
|   | |||||||
| @@ -1217,7 +1217,7 @@ redis.call("ZREMRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | |||||||
| return redis.status_reply("OK") | return redis.status_reply("OK") | ||||||
| `) | `) | ||||||
|  |  | ||||||
| // ReclaimStateAggregationSets checks for any stale aggregation sets in the given queue, and | // ReclaimStaleAggregationSets 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. | // reclaim tasks in the stale aggregation set by putting them back in the group. | ||||||
| func (r *RDB) ReclaimStaleAggregationSets(qname string) error { | func (r *RDB) ReclaimStaleAggregationSets(qname string) error { | ||||||
| 	var op errors.Op = "RDB.ReclaimStaleAggregationSets" | 	var op errors.Op = "RDB.ReclaimStaleAggregationSets" | ||||||
| @@ -1241,9 +1241,7 @@ return table.getn(ids)`) | |||||||
|  |  | ||||||
| // DeleteExpiredCompletedTasks checks for any expired tasks in the given queue's completed set, | // DeleteExpiredCompletedTasks checks for any expired tasks in the given queue's completed set, | ||||||
| // and delete all expired tasks. | // and delete all expired tasks. | ||||||
| func (r *RDB) DeleteExpiredCompletedTasks(qname string) error { | func (r *RDB) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||||
| 	// Note: Do this operation in fix batches to prevent long running script. |  | ||||||
| 	const batchSize = 100 |  | ||||||
| 	for { | 	for { | ||||||
| 		n, err := r.deleteExpiredCompletedTasks(qname, batchSize) | 		n, err := r.deleteExpiredCompletedTasks(qname, batchSize) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @@ -1284,7 +1282,10 @@ local res = {} | |||||||
| local ids = redis.call("ZRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | local ids = redis.call("ZRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | ||||||
| for _, id in ipairs(ids) do | for _, id in ipairs(ids) do | ||||||
| 	local key = ARGV[2] .. id | 	local key = ARGV[2] .. id | ||||||
| 	table.insert(res, redis.call("HGET", key, "msg")) | 	local v = redis.call("HGET", key, "msg") | ||||||
|  | 	if v then | ||||||
|  | 		table.insert(res, v) | ||||||
|  | 	end | ||||||
| end | end | ||||||
| return res | return res | ||||||
| `) | `) | ||||||
|   | |||||||
| @@ -2542,8 +2542,8 @@ func TestDeleteExpiredCompletedTasks(t *testing.T) { | |||||||
| 		h.FlushDB(t, r.client) | 		h.FlushDB(t, r.client) | ||||||
| 		h.SeedAllCompletedQueues(t, r.client, tc.completed) | 		h.SeedAllCompletedQueues(t, r.client, tc.completed) | ||||||
|  |  | ||||||
| 		if err := r.DeleteExpiredCompletedTasks(tc.qname); err != nil { | 		if err := r.DeleteExpiredCompletedTasks(tc.qname, 100); err != nil { | ||||||
| 			t.Errorf("DeleteExpiredCompletedTasks(%q) failed: %v", tc.qname, err) | 			t.Errorf("DeleteExpiredCompletedTasks(%q, 100) failed: %v", tc.qname, err) | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/redis/go-redis/v9" |  | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var errRedisDown = errors.New("testutil: redis is down") | var errRedisDown = errors.New("testutil: redis is down") | ||||||
| @@ -145,13 +145,13 @@ func (tb *TestBroker) ForwardIfReady(qnames ...string) error { | |||||||
| 	return tb.real.ForwardIfReady(qnames...) | 	return tb.real.ForwardIfReady(qnames...) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string) error { | func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||||
| 	tb.mu.Lock() | 	tb.mu.Lock() | ||||||
| 	defer tb.mu.Unlock() | 	defer tb.mu.Unlock() | ||||||
| 	if tb.sleeping { | 	if tb.sleeping { | ||||||
| 		return errRedisDown | 		return errRedisDown | ||||||
| 	} | 	} | ||||||
| 	return tb.real.DeleteExpiredCompletedTasks(qname) | 	return tb.real.DeleteExpiredCompletedTasks(qname, batchSize) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (tb *TestBroker) ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*base.TaskMessage, error) { | func (tb *TestBroker) ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*base.TaskMessage, error) { | ||||||
|   | |||||||
| @@ -27,6 +27,9 @@ type janitor struct { | |||||||
|  |  | ||||||
| 	// average interval between checks. | 	// average interval between checks. | ||||||
| 	avgInterval time.Duration | 	avgInterval time.Duration | ||||||
|  |  | ||||||
|  | 	// number of tasks to be deleted when janitor runs to delete the expired completed tasks. | ||||||
|  | 	batchSize int | ||||||
| } | } | ||||||
|  |  | ||||||
| type janitorParams struct { | type janitorParams struct { | ||||||
| @@ -34,6 +37,7 @@ type janitorParams struct { | |||||||
| 	broker    base.Broker | 	broker    base.Broker | ||||||
| 	queues    []string | 	queues    []string | ||||||
| 	interval  time.Duration | 	interval  time.Duration | ||||||
|  | 	batchSize int | ||||||
| } | } | ||||||
|  |  | ||||||
| func newJanitor(params janitorParams) *janitor { | func newJanitor(params janitorParams) *janitor { | ||||||
| @@ -43,6 +47,7 @@ func newJanitor(params janitorParams) *janitor { | |||||||
| 		done:        make(chan struct{}), | 		done:        make(chan struct{}), | ||||||
| 		queues:      params.queues, | 		queues:      params.queues, | ||||||
| 		avgInterval: params.interval, | 		avgInterval: params.interval, | ||||||
|  | 		batchSize:   params.batchSize, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -73,7 +78,7 @@ func (j *janitor) start(wg *sync.WaitGroup) { | |||||||
|  |  | ||||||
| func (j *janitor) exec() { | func (j *janitor) exec() { | ||||||
| 	for _, qname := range j.queues { | 	for _, qname := range j.queues { | ||||||
| 		if err := j.broker.DeleteExpiredCompletedTasks(qname); err != nil { | 		if err := j.broker.DeleteExpiredCompletedTasks(qname, j.batchSize); err != nil { | ||||||
| 			j.logger.Errorf("Failed to delete expired completed tasks from queue %q: %v", | 			j.logger.Errorf("Failed to delete expired completed tasks from queue %q: %v", | ||||||
| 				qname, err) | 				qname, err) | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -26,11 +26,13 @@ func TestJanitor(t *testing.T) { | |||||||
| 	defer r.Close() | 	defer r.Close() | ||||||
| 	rdbClient := rdb.NewRDB(r) | 	rdbClient := rdb.NewRDB(r) | ||||||
| 	const interval = 1 * time.Second | 	const interval = 1 * time.Second | ||||||
|  | 	const batchSize = 100 | ||||||
| 	janitor := newJanitor(janitorParams{ | 	janitor := newJanitor(janitorParams{ | ||||||
| 		logger:    testLogger, | 		logger:    testLogger, | ||||||
| 		broker:    rdbClient, | 		broker:    rdbClient, | ||||||
| 		queues:    []string{"default", "custom"}, | 		queues:    []string{"default", "custom"}, | ||||||
| 		interval:  interval, | 		interval:  interval, | ||||||
|  | 		batchSize: batchSize, | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	now := time.Now() | 	now := time.Now() | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								processor.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								processor.go
									
									
									
									
									
								
							| @@ -37,6 +37,7 @@ type processor struct { | |||||||
| 	// orderedQueues is set only in strict-priority mode. | 	// orderedQueues is set only in strict-priority mode. | ||||||
| 	orderedQueues []string | 	orderedQueues []string | ||||||
|  |  | ||||||
|  | 	taskCheckInterval time.Duration | ||||||
| 	retryDelayFunc    RetryDelayFunc | 	retryDelayFunc    RetryDelayFunc | ||||||
| 	isFailureFunc     func(error) bool | 	isFailureFunc     func(error) bool | ||||||
|  |  | ||||||
| @@ -76,6 +77,7 @@ type processorParams struct { | |||||||
| 	broker            base.Broker | 	broker            base.Broker | ||||||
| 	baseCtxFn         func() context.Context | 	baseCtxFn         func() context.Context | ||||||
| 	retryDelayFunc    RetryDelayFunc | 	retryDelayFunc    RetryDelayFunc | ||||||
|  | 	taskCheckInterval time.Duration | ||||||
| 	isFailureFunc     func(error) bool | 	isFailureFunc     func(error) bool | ||||||
| 	syncCh            chan<- *syncRequest | 	syncCh            chan<- *syncRequest | ||||||
| 	cancelations      *base.Cancelations | 	cancelations      *base.Cancelations | ||||||
| @@ -102,6 +104,7 @@ func newProcessor(params processorParams) *processor { | |||||||
| 		clock:             timeutil.NewRealClock(), | 		clock:             timeutil.NewRealClock(), | ||||||
| 		queueConfig:       queues, | 		queueConfig:       queues, | ||||||
| 		orderedQueues:     orderedQueues, | 		orderedQueues:     orderedQueues, | ||||||
|  | 		taskCheckInterval: params.taskCheckInterval, | ||||||
| 		retryDelayFunc:    params.retryDelayFunc, | 		retryDelayFunc:    params.retryDelayFunc, | ||||||
| 		isFailureFunc:     params.isFailureFunc, | 		isFailureFunc:     params.isFailureFunc, | ||||||
| 		syncRequestCh:     params.syncCh, | 		syncRequestCh:     params.syncCh, | ||||||
| @@ -178,7 +181,7 @@ func (p *processor) exec() { | |||||||
| 			// Sleep to avoid slamming redis and let scheduler move tasks into queues. | 			// Sleep to avoid slamming redis and let scheduler move tasks into queues. | ||||||
| 			// Note: We are not using blocking pop operation and polling queues instead. | 			// Note: We are not using blocking pop operation and polling queues instead. | ||||||
| 			// This adds significant load to redis. | 			// This adds significant load to redis. | ||||||
| 			time.Sleep(time.Second) | 			time.Sleep(p.taskCheckInterval) | ||||||
| 			<-p.sema // release token | 			<-p.sema // release token | ||||||
| 			return | 			return | ||||||
| 		case err != nil: | 		case err != nil: | ||||||
| @@ -412,21 +415,19 @@ func (p *processor) queues() []string { | |||||||
| func (p *processor) perform(ctx context.Context, task *Task) (err error) { | func (p *processor) perform(ctx context.Context, task *Task) (err error) { | ||||||
| 	defer func() { | 	defer func() { | ||||||
| 		if x := recover(); x != nil { | 		if x := recover(); x != nil { | ||||||
| 			errMsg := string(debug.Stack()) | 			p.logger.Errorf("recovering from panic. See the stack trace below for details:\n%s", string(debug.Stack())) | ||||||
|  |  | ||||||
| 			p.logger.Errorf("recovering from panic. See the stack trace below for details:\n%s", errMsg) |  | ||||||
| 			_, file, line, ok := runtime.Caller(1) // skip the first frame (panic itself) | 			_, file, line, ok := runtime.Caller(1) // skip the first frame (panic itself) | ||||||
| 			if ok && strings.Contains(file, "runtime/") { | 			if ok && strings.Contains(file, "runtime/") { | ||||||
| 				// The panic came from the runtime, most likely due to incorrect | 				// The panic came from the runtime, most likely due to incorrect | ||||||
| 				// map/slice usage. The parent frame should have the real trigger. | 				// map/slice usage. The parent frame should have the real trigger. | ||||||
| 				_, file, line, ok = runtime.Caller(2) | 				_, file, line, ok = runtime.Caller(2) | ||||||
| 			} | 			} | ||||||
|  | 			var errMsg string | ||||||
| 			// Include the file and line number info in the error, if runtime.Caller returned ok. | 			// Include the file and line number info in the error, if runtime.Caller returned ok. | ||||||
| 			if ok { | 			if ok { | ||||||
| 				err = fmt.Errorf("panic [%s:%d]: %v", file, line, x) | 				errMsg = fmt.Sprintf("panic [%s:%d]: %v", file, line, x) | ||||||
| 			} else { | 			} else { | ||||||
| 				err = fmt.Errorf("panic: %v", x) | 				errMsg = fmt.Sprintf("panic: %v", x) | ||||||
| 			} | 			} | ||||||
| 			err = &errors.PanicError{ | 			err = &errors.PanicError{ | ||||||
| 				ErrMsg: errMsg, | 				ErrMsg: errMsg, | ||||||
|   | |||||||
| @@ -67,6 +67,7 @@ func newProcessorForTest(t *testing.T, r *rdb.RDB, h Handler) *processor { | |||||||
| 		broker:            r, | 		broker:            r, | ||||||
| 		baseCtxFn:         context.Background, | 		baseCtxFn:         context.Background, | ||||||
| 		retryDelayFunc:    DefaultRetryDelayFunc, | 		retryDelayFunc:    DefaultRetryDelayFunc, | ||||||
|  | 		taskCheckInterval: defaultTaskCheckInterval, | ||||||
| 		isFailureFunc:     defaultIsFailureFunc, | 		isFailureFunc:     defaultIsFailureFunc, | ||||||
| 		syncCh:            syncCh, | 		syncCh:            syncCh, | ||||||
| 		cancelations:      base.NewCancelations(), | 		cancelations:      base.NewCancelations(), | ||||||
| @@ -542,6 +543,7 @@ func TestProcessorWithExpiredLease(t *testing.T) { | |||||||
| 			logger:            testLogger, | 			logger:            testLogger, | ||||||
| 			broker:            rdbClient, | 			broker:            rdbClient, | ||||||
| 			baseCtxFn:         context.Background, | 			baseCtxFn:         context.Background, | ||||||
|  | 			taskCheckInterval: defaultTaskCheckInterval, | ||||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||||
| 			isFailureFunc:     defaultIsFailureFunc, | 			isFailureFunc:     defaultIsFailureFunc, | ||||||
| 			syncCh:            syncCh, | 			syncCh:            syncCh, | ||||||
| @@ -696,6 +698,7 @@ func TestProcessorWithStrictPriority(t *testing.T) { | |||||||
| 			logger:            testLogger, | 			logger:            testLogger, | ||||||
| 			broker:            rdbClient, | 			broker:            rdbClient, | ||||||
| 			baseCtxFn:         context.Background, | 			baseCtxFn:         context.Background, | ||||||
|  | 			taskCheckInterval: defaultTaskCheckInterval, | ||||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||||
| 			isFailureFunc:     defaultIsFailureFunc, | 			isFailureFunc:     defaultIsFailureFunc, | ||||||
| 			syncCh:            syncCh, | 			syncCh:            syncCh, | ||||||
|   | |||||||
							
								
								
									
										24
									
								
								scheduler.go
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								scheduler.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/redis/go-redis/v9" |  | ||||||
| 	"github.com/google/uuid" | 	"github.com/google/uuid" | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	"github.com/hibiken/asynq/internal/log" | 	"github.com/hibiken/asynq/internal/log" | ||||||
| 	"github.com/hibiken/asynq/internal/rdb" | 	"github.com/hibiken/asynq/internal/rdb" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| 	"github.com/robfig/cron/v3" | 	"github.com/robfig/cron/v3" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -43,15 +43,27 @@ type Scheduler struct { | |||||||
| 	// to avoid using cron.EntryID as the public API of | 	// to avoid using cron.EntryID as the public API of | ||||||
| 	// the Scheduler. | 	// the Scheduler. | ||||||
| 	idmap map[string]cron.EntryID | 	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. | // 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 | // The parameter opts is optional, defaults will be used if opts is set to nil | ||||||
| func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | ||||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||||
| 	if !ok { | 	if !ok { | ||||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | 		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 { | 	if opts == nil { | ||||||
| 		opts = &SchedulerOpts{} | 		opts = &SchedulerOpts{} | ||||||
| 	} | 	} | ||||||
| @@ -72,7 +84,7 @@ func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | |||||||
| 		id:              generateSchedulerID(), | 		id:              generateSchedulerID(), | ||||||
| 		state:           &serverState{value: srvStateNew}, | 		state:           &serverState{value: srvStateNew}, | ||||||
| 		logger:          logger, | 		logger:          logger, | ||||||
| 		client:          NewClient(r), | 		client:          NewClientFromRedisClient(c), | ||||||
| 		rdb:             rdb.NewRDB(c), | 		rdb:             rdb.NewRDB(c), | ||||||
| 		cron:            cron.New(cron.WithLocation(loc)), | 		cron:            cron.New(cron.WithLocation(loc)), | ||||||
| 		location:        loc, | 		location:        loc, | ||||||
| @@ -261,8 +273,12 @@ func (s *Scheduler) Shutdown() { | |||||||
| 	s.wg.Wait() | 	s.wg.Wait() | ||||||
|  |  | ||||||
| 	s.clearHistory() | 	s.clearHistory() | ||||||
| 	s.client.Close() | 	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.rdb.Close() | ||||||
|  | 	} | ||||||
| 	s.logger.Info("Scheduler stopped") | 	s.logger.Info("Scheduler stopped") | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| package asynq | package asynq | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| 	"sync" | 	"sync" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
| @@ -58,6 +59,7 @@ func TestSchedulerRegister(t *testing.T) { | |||||||
|  |  | ||||||
| 	r := setup(t) | 	r := setup(t) | ||||||
|  |  | ||||||
|  | 	// Tests for new redis connection. | ||||||
| 	for _, tc := range tests { | 	for _, tc := range tests { | ||||||
| 		scheduler := NewScheduler(getRedisConnOpt(t), nil) | 		scheduler := NewScheduler(getRedisConnOpt(t), nil) | ||||||
| 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | ||||||
| @@ -75,6 +77,28 @@ func TestSchedulerRegister(t *testing.T) { | |||||||
| 			t.Errorf("mismatch found in queue %q: (-want,+got)\n%s", tc.queue, diff) | 			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) { | func TestSchedulerWhenRedisDown(t *testing.T) { | ||||||
|   | |||||||
							
								
								
									
										69
									
								
								server.go
									
									
									
									
									
								
							
							
						
						
									
										69
									
								
								server.go
									
									
									
									
									
								
							| @@ -15,10 +15,10 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/redis/go-redis/v9" |  | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	"github.com/hibiken/asynq/internal/log" | 	"github.com/hibiken/asynq/internal/log" | ||||||
| 	"github.com/hibiken/asynq/internal/rdb" | 	"github.com/hibiken/asynq/internal/rdb" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Server is responsible for task processing and task lifecycle management. | // Server is responsible for task processing and task lifecycle management. | ||||||
| @@ -37,6 +37,9 @@ type Server struct { | |||||||
| 	logger *log.Logger | 	logger *log.Logger | ||||||
|  |  | ||||||
| 	broker base.Broker | 	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 | 	state *serverState | ||||||
|  |  | ||||||
| @@ -104,6 +107,15 @@ type Config struct { | |||||||
| 	// If this is defined, then it MUST return a non-nil context | 	// If this is defined, then it MUST return a non-nil context | ||||||
| 	BaseContext func() context.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. | 	// Function to calculate retry delay for a failed task. | ||||||
| 	// | 	// | ||||||
| 	// By default, it uses exponential backoff algorithm to calculate the delay. | 	// By default, it uses exponential backoff algorithm to calculate the delay. | ||||||
| @@ -230,6 +242,17 @@ type Config struct { | |||||||
| 	// | 	// | ||||||
| 	// If unset or nil, the group aggregation feature will be disabled on the server. | 	// If unset or nil, the group aggregation feature will be disabled on the server. | ||||||
| 	GroupAggregator GroupAggregator | 	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. | // GroupAggregator aggregates a group of tasks into one before the tasks are passed to the Handler. | ||||||
| @@ -390,6 +413,8 @@ var defaultQueueConfig = map[string]int{ | |||||||
| } | } | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
|  | 	defaultTaskCheckInterval = 1 * time.Second | ||||||
|  |  | ||||||
| 	defaultShutdownTimeout = 8 * time.Second | 	defaultShutdownTimeout = 8 * time.Second | ||||||
|  |  | ||||||
| 	defaultHealthCheckInterval = 15 * time.Second | 	defaultHealthCheckInterval = 15 * time.Second | ||||||
| @@ -397,15 +422,28 @@ const ( | |||||||
| 	defaultDelayedTaskCheckInterval = 5 * time.Second | 	defaultDelayedTaskCheckInterval = 5 * time.Second | ||||||
|  |  | ||||||
| 	defaultGroupGracePeriod = 1 * time.Minute | 	defaultGroupGracePeriod = 1 * time.Minute | ||||||
|  |  | ||||||
|  | 	defaultJanitorInterval = 8 * time.Second | ||||||
|  |  | ||||||
|  | 	defaultJanitorBatchSize = 100 | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // NewServer returns a new Server given a redis connection option | // NewServer returns a new Server given a redis connection option | ||||||
| // and server configuration. | // and server configuration. | ||||||
| func NewServer(r RedisConnOpt, cfg Config) *Server { | func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||||
| 	if !ok { | 	if !ok { | ||||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | 		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 | 	baseCtxFn := cfg.BaseContext | ||||||
| 	if baseCtxFn == nil { | 	if baseCtxFn == nil { | ||||||
| 		baseCtxFn = context.Background | 		baseCtxFn = context.Background | ||||||
| @@ -414,6 +452,12 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | |||||||
| 	if n < 1 { | 	if n < 1 { | ||||||
| 		n = runtime.NumCPU() | 		n = runtime.NumCPU() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	taskCheckInterval := cfg.TaskCheckInterval | ||||||
|  | 	if taskCheckInterval <= 0 { | ||||||
|  | 		taskCheckInterval = defaultTaskCheckInterval | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	delayFunc := cfg.RetryDelayFunc | 	delayFunc := cfg.RetryDelayFunc | ||||||
| 	if delayFunc == nil { | 	if delayFunc == nil { | ||||||
| 		delayFunc = DefaultRetryDelayFunc | 		delayFunc = DefaultRetryDelayFunc | ||||||
| @@ -503,6 +547,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | |||||||
| 		logger:            logger, | 		logger:            logger, | ||||||
| 		broker:            rdb, | 		broker:            rdb, | ||||||
| 		retryDelayFunc:    delayFunc, | 		retryDelayFunc:    delayFunc, | ||||||
|  | 		taskCheckInterval: taskCheckInterval, | ||||||
| 		baseCtxFn:         baseCtxFn, | 		baseCtxFn:         baseCtxFn, | ||||||
| 		isFailureFunc:     isFailureFunc, | 		isFailureFunc:     isFailureFunc, | ||||||
| 		syncCh:            syncCh, | 		syncCh:            syncCh, | ||||||
| @@ -529,11 +574,26 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | |||||||
| 		interval:        healthcheckInterval, | 		interval:        healthcheckInterval, | ||||||
| 		healthcheckFunc: cfg.HealthCheckFunc, | 		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{ | 	janitor := newJanitor(janitorParams{ | ||||||
| 		logger:    logger, | 		logger:    logger, | ||||||
| 		broker:    rdb, | 		broker:    rdb, | ||||||
| 		queues:    qnames, | 		queues:    qnames, | ||||||
| 		interval: 8 * time.Second, | 		interval:  janitorInterval, | ||||||
|  | 		batchSize: janitorBatchSize, | ||||||
| 	}) | 	}) | ||||||
| 	aggregator := newAggregator(aggregatorParams{ | 	aggregator := newAggregator(aggregatorParams{ | ||||||
| 		logger:          logger, | 		logger:          logger, | ||||||
| @@ -547,6 +607,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | |||||||
| 	return &Server{ | 	return &Server{ | ||||||
| 		logger:           logger, | 		logger:           logger, | ||||||
| 		broker:           rdb, | 		broker:           rdb, | ||||||
|  | 		sharedConnection: true, | ||||||
| 		state:            srvState, | 		state:            srvState, | ||||||
| 		forwarder:        forwarder, | 		forwarder:        forwarder, | ||||||
| 		processor:        processor, | 		processor:        processor, | ||||||
| @@ -684,7 +745,9 @@ func (srv *Server) Shutdown() { | |||||||
| 	srv.heartbeater.shutdown() | 	srv.heartbeater.shutdown() | ||||||
| 	srv.wg.Wait() | 	srv.wg.Wait() | ||||||
|  |  | ||||||
|  | 	if !srv.sharedConnection { | ||||||
| 		srv.broker.Close() | 		srv.broker.Close() | ||||||
|  | 	} | ||||||
| 	srv.logger.Info("Exiting") | 	srv.logger.Info("Exiting") | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,22 +14,12 @@ import ( | |||||||
| 	"github.com/hibiken/asynq/internal/rdb" | 	"github.com/hibiken/asynq/internal/rdb" | ||||||
| 	"github.com/hibiken/asynq/internal/testbroker" | 	"github.com/hibiken/asynq/internal/testbroker" | ||||||
| 	"github.com/hibiken/asynq/internal/testutil" | 	"github.com/hibiken/asynq/internal/testutil" | ||||||
|  |  | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| 	"go.uber.org/goleak" | 	"go.uber.org/goleak" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestServer(t *testing.T) { | func testServer(t *testing.T, c *Client, srv *Server) { | ||||||
| 	// 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 | 	// no-op handler | ||||||
| 	h := func(ctx context.Context, task *Task) error { | 	h := func(ctx context.Context, task *Task) error { | ||||||
| 		return nil | 		return nil | ||||||
| @@ -53,6 +43,43 @@ func TestServer(t *testing.T) { | |||||||
| 	srv.Shutdown() | 	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) { | func TestServerRun(t *testing.T) { | ||||||
| 	// https://github.com/go-redis/redis/issues/1029 | 	// https://github.com/go-redis/redis/issues/1029 | ||||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| // +build linux bsd darwin | //go:build linux || dragonfly || freebsd || netbsd || openbsd || darwin | ||||||
|  |  | ||||||
| package asynq | package asynq | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| // +build windows | //go:build windows | ||||||
|  |  | ||||||
| package asynq | 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: | In order to use the tool, compile it using the following command: | ||||||
|  |  | ||||||
|     go install github.com/hibiken/asynq/tools/asynq |     go install github.com/hibiken/asynq/tools/asynq@latest | ||||||
|  |  | ||||||
| This will create the asynq executable under your `$GOPATH/bin` directory. | 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 dash` | ||||||
| - `asynq stats` | - `asynq stats` | ||||||
| - `asynq queue [ls inspect history rm pause unpause]` | - `asynq queue [ls inspect history rm pause unpause]` | ||||||
| - `asynq task [ls cancel delete archive run delete-all archive-all run-all]` | - `asynq task [ls cancel delete archive run deleteall archiveall runall]` | ||||||
| - `asynq server [ls]` | - `asynq server [ls]` | ||||||
|  |  | ||||||
| ### Global flags | ### Global flags | ||||||
|   | |||||||
| @@ -369,7 +369,7 @@ func createRDB() *rdb.RDB { | |||||||
| 	return rdb.NewRDB(c) | 	return rdb.NewRDB(c) | ||||||
| } | } | ||||||
|  |  | ||||||
| // createRDB creates a Inspector instance using flag values and returns it. | // createInspector creates a Inspector instance using flag values and returns it. | ||||||
| func createInspector() *asynq.Inspector { | func createInspector() *asynq.Inspector { | ||||||
| 	return asynq.NewInspector(getRedisConnOpt()) | 	return asynq.NewInspector(getRedisConnOpt()) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| module github.com/hibiken/asynq/tools | module github.com/hibiken/asynq/tools | ||||||
|  |  | ||||||
| go 1.18 | go 1.20 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/MakeNowJust/heredoc/v2 v2.0.1 | 	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/hibiken/asynq/x v0.0.0-20220131170841-349f4c50fb1d | ||||||
| 	github.com/mattn/go-runewidth v0.0.13 | 	github.com/mattn/go-runewidth v0.0.13 | ||||||
| 	github.com/mitchellh/go-homedir v1.1.0 | 	github.com/mitchellh/go-homedir v1.1.0 | ||||||
| 	github.com/prometheus/client_golang v1.11.0 | 	github.com/prometheus/client_golang v1.11.1 | ||||||
| 	github.com/redis/go-redis/v9 v9.0.5 | 	github.com/redis/go-redis/v9 v9.0.5 | ||||||
| 	github.com/spf13/cobra v1.1.1 | 	github.com/spf13/cobra v1.1.1 | ||||||
| 	github.com/spf13/pflag v1.0.5 | 	github.com/spf13/pflag v1.0.5 | ||||||
| @@ -47,7 +47,7 @@ require ( | |||||||
| 	github.com/subosito/gotenv v1.2.0 // indirect | 	github.com/subosito/gotenv v1.2.0 // indirect | ||||||
| 	golang.org/x/sys v0.10.0 // indirect | 	golang.org/x/sys v0.10.0 // indirect | ||||||
| 	golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf // indirect | 	golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf // indirect | ||||||
| 	golang.org/x/text v0.3.7 // indirect | 	golang.org/x/text v0.3.8 // indirect | ||||||
| 	golang.org/x/time v0.3.0 // indirect | 	golang.org/x/time v0.3.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.31.0 // indirect | 	google.golang.org/protobuf v1.31.0 // indirect | ||||||
| 	gopkg.in/ini.v1 v1.51.0 // indirect | 	gopkg.in/ini.v1 v1.51.0 // indirect | ||||||
|   | |||||||
| @@ -235,8 +235,9 @@ 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 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.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.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.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-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-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.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||||
| @@ -423,8 +424,9 @@ 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.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.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||||
| golang.org/x/text v0.3.6/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.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-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.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||||
| golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= | golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4= | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								x/go.mod
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								x/go.mod
									
									
									
									
									
								
							| @@ -1,10 +1,26 @@ | |||||||
| module github.com/hibiken/asynq/x | module github.com/hibiken/asynq/x | ||||||
|  |  | ||||||
| go 1.16 | go 1.20 | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
| 	github.com/go-redis/redis/v8 v8.11.4 | 	github.com/google/uuid v1.4.0 | ||||||
| 	github.com/google/uuid v1.3.0 | 	github.com/hibiken/asynq v0.24.1 | ||||||
| 	github.com/hibiken/asynq v0.21.0 | 	github.com/prometheus/client_golang v1.11.1 | ||||||
| 	github.com/prometheus/client_golang v1.11.0 | 	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 | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										147
									
								
								x/go.sum
									
									
									
									
									
								
							
							
						
						
									
										147
									
								
								x/go.sum
									
									
									
									
									
								
							| @@ -1,6 +1,4 @@ | |||||||
| 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= | 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-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||||
| github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/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= | github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||||
| @@ -10,35 +8,26 @@ 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.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= | ||||||
| github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= | 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/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= | ||||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | 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/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||||
| github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||||
| github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | github.com/cespare/xxhash/v2 v2.2.0/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.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/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 h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||||
| 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.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||||
| github.com/go-kit/kit v0.9.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-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.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.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | 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-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/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.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.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
| @@ -47,28 +36,25 @@ 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.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-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.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.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||||
| github.com/golang/protobuf v1.4.3/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.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.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | 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.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | 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.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.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.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.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.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/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.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||||
| github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= | github.com/google/uuid v1.4.0 h1:MtMxsa51/r9yyhkyLsVeVt0B+BGQZzpQiTQ4eHZ8bc4= | ||||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | github.com/google/uuid v1.4.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||||
| github.com/hibiken/asynq v0.21.0 h1:uH9XogJhjq/S39E0/DEPWLZQ6hHJ73UiblZTe4RzHwA= | github.com/hibiken/asynq v0.24.1 h1:+5iIEAyA9K/lcSPvx3qoPtsKJeKI5u9aOIvUmSsazEw= | ||||||
| github.com/hibiken/asynq v0.21.0/go.mod h1:tyc63ojaW8SJ5SBm8mvI4DDONsguP5HE85EEl4Qr5Ig= | github.com/hibiken/asynq v0.24.1/go.mod h1:u5qVeSbrnfT+vtG5Mq8ZPzQu/BmCKMHvTGb91uy9Tts= | ||||||
| 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/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.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||||
| github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||||
| @@ -79,8 +65,10 @@ 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/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/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.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/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.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 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= | ||||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | 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= | github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||||
| @@ -89,32 +77,17 @@ 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/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-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/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.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
| github.com/pkg/errors v0.8.1/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/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/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 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.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.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.1 h1:+4eQaD7vAZ6DsfsxB15hbE0odUjGI5ARs9yskGu1v4s= | ||||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | 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-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-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 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= | ||||||
| github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||||
| github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||||
| @@ -126,133 +99,99 @@ 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.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 h1:mxy4L2jP6qMonqmq+aTtOx1ifVWUgG/TAmntgbh3xv4= | ||||||
| github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | 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 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= | ||||||
| github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | 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.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||||
| github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= | 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/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.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.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/objx v0.1.1/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.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | 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.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||||
| github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= | ||||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | ||||||
| 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-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-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-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||||
| golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||||
| 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-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-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-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-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-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-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-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-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||||
| 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/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-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-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-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-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-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/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-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-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-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-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-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-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-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-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-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-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40 h1:JWgyZ1qgdTaF3N3oxC+MdTV7qvEEgHo3otj+HB5CM7Q= | golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | 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/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.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.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.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.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-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-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.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | 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-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-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-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= | 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/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-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-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 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.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.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.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-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.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/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 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/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.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.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.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.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.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= | 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" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/go-redis/redis/v8" |  | ||||||
| 	"github.com/hibiken/asynq" | 	"github.com/hibiken/asynq" | ||||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | 	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. | // NewSemaphore creates a counting Semaphore for the given scope with the given number of tokens. | ||||||
|   | |||||||
| @@ -8,11 +8,11 @@ import ( | |||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/go-redis/redis/v8" |  | ||||||
| 	"github.com/google/uuid" | 	"github.com/google/uuid" | ||||||
| 	"github.com/hibiken/asynq" | 	"github.com/hibiken/asynq" | ||||||
| 	"github.com/hibiken/asynq/internal/base" | 	"github.com/hibiken/asynq/internal/base" | ||||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||||
|  | 	"github.com/redis/go-redis/v9" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -91,7 +91,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | |||||||
| 			maxConcurrency: 3, | 			maxConcurrency: 3, | ||||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | ||||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||||
| 				return asynqcontext.New(&base.TaskMessage{ | 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 					ID:    id, | 					ID:    id, | ||||||
| 					Queue: "task-1", | 					Queue: "task-1", | ||||||
| 				}, time.Now().Add(time.Second)) | 				}, time.Now().Add(time.Second)) | ||||||
| @@ -104,7 +104,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | |||||||
| 			maxConcurrency: 3, | 			maxConcurrency: 3, | ||||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString(), uuid.NewString(), uuid.NewString()}, | 			taskIDs:        []string{uuid.NewString(), uuid.NewString(), uuid.NewString(), uuid.NewString()}, | ||||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||||
| 				return asynqcontext.New(&base.TaskMessage{ | 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 					ID:    id, | 					ID:    id, | ||||||
| 					Queue: "task-2", | 					Queue: "task-2", | ||||||
| 				}, time.Now().Add(time.Second)) | 				}, 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, | 	// 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 | 	// 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()), | 		Score:  float64(time.Now().Add(-10 * time.Second).Unix()), | ||||||
| 		Member: taskID, | 		Member: taskID, | ||||||
| 	}) | 	}) | ||||||
| @@ -219,7 +219,7 @@ func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | |||||||
| 	sema := NewSemaphore(opt, "stale-token", 1) | 	sema := NewSemaphore(opt, "stale-token", 1) | ||||||
| 	defer sema.Close() | 	defer sema.Close() | ||||||
|  |  | ||||||
| 	ctx, cancel := asynqcontext.New(&base.TaskMessage{ | 	ctx, cancel := asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 		ID:    taskID, | 		ID:    taskID, | ||||||
| 		Queue: "task-1", | 		Queue: "task-1", | ||||||
| 	}, time.Now().Add(time.Second)) | 	}, time.Now().Add(time.Second)) | ||||||
| @@ -248,7 +248,7 @@ func TestNewSemaphore_Release(t *testing.T) { | |||||||
| 			name:    "task-5", | 			name:    "task-5", | ||||||
| 			taskIDs: []string{uuid.NewString()}, | 			taskIDs: []string{uuid.NewString()}, | ||||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||||
| 				return asynqcontext.New(&base.TaskMessage{ | 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 					ID:    id, | 					ID:    id, | ||||||
| 					Queue: "task-3", | 					Queue: "task-3", | ||||||
| 				}, time.Now().Add(time.Second)) | 				}, time.Now().Add(time.Second)) | ||||||
| @@ -259,7 +259,7 @@ func TestNewSemaphore_Release(t *testing.T) { | |||||||
| 			name:    "task-6", | 			name:    "task-6", | ||||||
| 			taskIDs: []string{uuid.NewString(), uuid.NewString()}, | 			taskIDs: []string{uuid.NewString(), uuid.NewString()}, | ||||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||||
| 				return asynqcontext.New(&base.TaskMessage{ | 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 					ID:    id, | 					ID:    id, | ||||||
| 					Queue: "task-4", | 					Queue: "task-4", | ||||||
| 				}, time.Now().Add(time.Second)) | 				}, 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) | 				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++ { | 			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()), | 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||||
| 					Member: tt.taskIDs[i], | 					Member: tt.taskIDs[i], | ||||||
| 				}) | 				}) | ||||||
| @@ -337,7 +337,7 @@ func TestNewSemaphore_Release_Error(t *testing.T) { | |||||||
| 			name:    "task-8", | 			name:    "task-8", | ||||||
| 			taskIDs: []string{uuid.NewString()}, | 			taskIDs: []string{uuid.NewString()}, | ||||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||||
| 				return asynqcontext.New(&base.TaskMessage{ | 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||||
| 					ID:    testID, | 					ID:    testID, | ||||||
| 					Queue: "task-4", | 					Queue: "task-4", | ||||||
| 				}, time.Now().Add(time.Second)) | 				}, 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) | 				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++ { | 			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()), | 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||||
| 					Member: tt.taskIDs[i], | 					Member: tt.taskIDs[i], | ||||||
| 				}) | 				}) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user