mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-20 21:26:14 +08:00 
			
		
		
		
	Compare commits
	
		
			89 Commits
		
	
	
		
			go-version
			...
			sohail/uni
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 71c746d00a | ||
|  | 02907551b4 | ||
|  | 127fac2e90 | ||
|  | 106c07adaa | ||
|  | 1c7195ff1a | ||
|  | 12cbba4926 | ||
|  | 80479b528d | ||
|  | e14c312fe3 | ||
|  | ad1f587403 | ||
|  | 8b32b38fd5 | ||
|  | 96a84fac0c | ||
|  | d2c207fbb8 | ||
|  | 1a7c61ac49 | ||
|  | 87375b5534 | ||
|  | ffd75ebb5f | ||
|  | d64fd328cb | ||
|  | 4f00f52c1d | ||
|  | 580d69e88f | ||
|  | c97652d408 | ||
|  | 4644d37ef4 | ||
|  | 45c0fc6ad9 | ||
|  | fd3eb86d95 | ||
|  | 3dbda60333 | ||
|  | 02c6dae7eb | ||
|  | 5cfcb71139 | ||
|  | c78e7b0ccd | ||
|  | b4db174032 | ||
|  | 39f1d8c3e6 | ||
|  | e70de721b8 | ||
|  | 6c06ad7e45 | ||
|  | a676d3d2fa | ||
|  | ef0d32965f | ||
|  | f16f9ac440 | ||
|  | 63f7cb7b17 | ||
|  | 04b3a3475d | ||
|  | 013190b824 | ||
|  | 1e102a5392 | ||
|  | e1a8a366a6 | ||
|  | c8c8adfaa6 | ||
|  | 03f4799712 | ||
|  | 3f4e211a3b | ||
|  | 0655c569f5 | ||
|  | 95a0768ae0 | ||
|  | f4b56498f2 | ||
|  | ae478d5b22 | ||
|  | ff7ef48463 | ||
|  | b1e13893ff | ||
|  | 0dc670d7d8 | ||
|  | 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 | ||
|  | 2165ed133b | ||
|  | 551b0c7119 | ||
|  | 123d560a44 | ||
|  | 5bef53d1ac | ||
|  | 90af7749ca | ||
|  | e4b8663154 | 
							
								
								
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +1,4 @@ | ||||
| # These are supported funding model platforms | ||||
|  | ||||
| github: [hibiken] # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] | ||||
| patreon: # Replace with a single Patreon username | ||||
| open_collective: # Replace with a single Open Collective username | ||||
| ko_fi: # Replace with a single Ko-fi username | ||||
| tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel | ||||
| community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry | ||||
| liberapay: # Replace with a single Liberapay username | ||||
| issuehunt: # Replace with a single IssueHunt username | ||||
| otechie: # Replace with a single Otechie username | ||||
| custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] | ||||
| github: [hibiken] | ||||
| open_collective: ken-hibino | ||||
|   | ||||
							
								
								
									
										15
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -3,13 +3,20 @@ name: Bug report | ||||
| about: Create a report to help us improve | ||||
| title: "[BUG] Description of the bug" | ||||
| labels: bug | ||||
| assignees: hibiken | ||||
|  | ||||
| assignees: | ||||
|   - hibiken | ||||
|   - kamikazechaser | ||||
|    | ||||
| --- | ||||
|  | ||||
| **Describe the bug** | ||||
| A clear and concise description of what the bug is. | ||||
|  | ||||
| **Environment (please complete the following information):** | ||||
|  - OS: [e.g. MacOS, Linux] | ||||
|  - `asynq` package version [e.g. v0.25.0] | ||||
|  - Redis/Valkey version  | ||||
|  | ||||
| **To Reproduce** | ||||
| Steps to reproduce the behavior (Code snippets if applicable): | ||||
| 1. Setup background processing ... | ||||
| @@ -22,9 +29,5 @@ A clear and concise description of what you expected to happen. | ||||
| **Screenshots** | ||||
| If applicable, add screenshots to help explain your problem. | ||||
|  | ||||
| **Environment (please complete the following information):** | ||||
|  - OS: [e.g. MacOS, Linux] | ||||
|  - Version of `asynq` package [e.g. v1.0.0] | ||||
|  | ||||
| **Additional context** | ||||
| Add any other context about the problem here. | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
								
							| @@ -3,7 +3,9 @@ name: Feature request | ||||
| about: Suggest an idea for this project | ||||
| title: "[FEATURE REQUEST] Description of the feature request" | ||||
| labels: enhancement | ||||
| assignees: hibiken | ||||
| assignees: | ||||
|   - hibiken | ||||
|   - kamikazechaser | ||||
|  | ||||
| --- | ||||
|  | ||||
|   | ||||
							
								
								
									
										24
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.github/dependabot.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| 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" | ||||
|   - package-ecosystem: "github-actions" | ||||
|     directory: "/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
							
								
								
									
										32
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,20 +11,20 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         image: redis:7 | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: 1.16.x | ||||
|           go-version: 1.23.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a new.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|           path: new.txt | ||||
| @@ -33,22 +33,22 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         image: redis:7 | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           ref: master | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: 1.15.x | ||||
|           go-version: 1.23.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a old.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: bench-current | ||||
|           path: old.txt | ||||
| @@ -58,25 +58,25 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: 1.15.x | ||||
|           go-version: 1.23.x | ||||
|       - name: Install benchstat | ||||
|         run: go get -u golang.org/x/perf/cmd/benchstat | ||||
|       - name: Download Incoming | ||||
|         uses: actions/download-artifact@v2 | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|       - name: Download Current | ||||
|         uses: actions/download-artifact@v2 | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: bench-current | ||||
|       - name: Benchstat Results | ||||
|         run: benchstat old.txt new.txt | tee -a benchstat.txt | ||||
|       - name: Upload benchstat results | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: benchstat | ||||
|           path: benchstat.txt | ||||
|   | ||||
							
								
								
									
										34
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										34
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,20 +7,21 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         go-version: [1.20.x, 1.21.x] | ||||
|         go-version: [1.22.x, 1.23.x] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         image: redis:7 | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go-version }} | ||||
|           cache: false | ||||
|  | ||||
|       - name: Build core module | ||||
|         run: go build -v ./... | ||||
| @@ -38,26 +39,27 @@ jobs: | ||||
|         run: go test -run=^$ -bench=. -loglevel=debug ./... | ||||
|  | ||||
|       - name: Upload coverage to Codecov | ||||
|         uses: codecov/codecov-action@v1 | ||||
|         uses: codecov/codecov-action@v5 | ||||
|  | ||||
|   build-tool: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         go-version: [1.18.x] | ||||
|         go-version: [1.22.x, 1.23.x] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         image: redis:7 | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go-version }} | ||||
|           cache: false | ||||
|  | ||||
|       - name: Build tools module | ||||
|         run: cd tools && go build -v ./... && cd .. | ||||
| @@ -65,3 +67,17 @@ jobs: | ||||
|       - name: Test tools module | ||||
|         run: cd tools && go test -race -v ./... && cd .. | ||||
|  | ||||
|   golangci: | ||||
|     name: lint | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: stable | ||||
|  | ||||
|       - name: golangci-lint | ||||
|         uses: golangci/golangci-lint-action@v6 | ||||
|         with: | ||||
|           version: v1.61 | ||||
|   | ||||
							
								
								
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -7,6 +7,34 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | ||||
|  | ||||
| ## [Unreleased] | ||||
|  | ||||
| ## [0.25.0] - 2024-10-29 | ||||
|  | ||||
| ### Upgrades | ||||
| - Minumum go version is set to 1.22 (PR: https://github.com/hibiken/asynq/pull/925) | ||||
| - Internal protobuf package is upgraded to address security advisories (PR: https://github.com/hibiken/asynq/pull/925) | ||||
| - Most packages are upgraded | ||||
| - CI/CD spec upgraded | ||||
|  | ||||
| ### Added | ||||
| - `IsPanicError` function is introduced to support catching of panic errors when processing tasks (PR: https://github.com/hibiken/asynq/pull/491) | ||||
| - `JanitorInterval` and `JanitorBatchSize` are added as Server options (PR: https://github.com/hibiken/asynq/pull/715) | ||||
| - `NewClientFromRedisClient` is introduced to allow reusing an existing redis client (PR: https://github.com/hibiken/asynq/pull/742) | ||||
| - `TaskCheckInterval` config option is added to specify the interval between checks for new tasks to process when all queues are empty (PR: https://github.com/hibiken/asynq/pull/694) | ||||
| - `Ping` method is added to Client, Server and Scheduler ((PR: https://github.com/hibiken/asynq/pull/585)) | ||||
| - `RevokeTask` error type is introduced to prevent a task from being retried or archived (PR: https://github.com/hibiken/asynq/pull/882) | ||||
| - `SentinelUsername` is added as a redis config option (PR: https://github.com/hibiken/asynq/pull/924) | ||||
| - Some jitter is introduced to improve latency when fetching jobs in the processor (PR: https://github.com/hibiken/asynq/pull/868) | ||||
| - Add task enqueue command to the CLI (PR: https://github.com/hibiken/asynq/pull/918) | ||||
| - Add a map cache (concurrent safe) to keep track of queues that ultimately reduces redis load when enqueuing tasks (PR: https://github.com/hibiken/asynq/pull/946) | ||||
|  | ||||
| ### Fixes | ||||
| - Archived tasks that are trimmed should now be deleted (PR: https://github.com/hibiken/asynq/pull/743) | ||||
| - Fix lua script when listing task messages with an expired lease (PR: https://github.com/hibiken/asynq/pull/709) | ||||
| - Fix potential context leaks due to cancellation not being called (PR: https://github.com/hibiken/asynq/pull/926) | ||||
| - Misc documentation fixes | ||||
| - Misc test fixes | ||||
|  | ||||
|  | ||||
| ## [0.24.1] - 2023-05-01 | ||||
|  | ||||
| ### Changed | ||||
|   | ||||
							
								
								
									
										6
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								Makefile
									
									
									
									
									
								
							| @@ -4,4 +4,8 @@ proto: internal/proto/asynq.proto | ||||
| 	protoc -I=$(ROOT_DIR)/internal/proto \ | ||||
| 				 --go_out=$(ROOT_DIR)/internal/proto \ | ||||
| 				 --go_opt=module=github.com/hibiken/asynq/internal/proto \ | ||||
| 				 $(ROOT_DIR)/internal/proto/asynq.proto | ||||
| 				 $(ROOT_DIR)/internal/proto/asynq.proto | ||||
|  | ||||
| .PHONY: lint | ||||
| lint: | ||||
| 	golangci-lint run | ||||
|   | ||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							| @@ -37,7 +37,6 @@ Task queues are used as a mechanism to distribute work across multiple machines. | ||||
| - [Flexible handler interface with support for middlewares](https://github.com/hibiken/asynq/wiki/Handler-Deep-Dive) | ||||
| - [Ability to pause queue](/tools/asynq/README.md#pause) to stop processing tasks from the queue | ||||
| - [Periodic Tasks](https://github.com/hibiken/asynq/wiki/Periodic-Tasks) | ||||
| - [Support Redis Cluster](https://github.com/hibiken/asynq/wiki/Redis-Cluster) for automatic sharding and high availability | ||||
| - [Support Redis Sentinels](https://github.com/hibiken/asynq/wiki/Automatic-Failover) for high availability | ||||
| - Integration with [Prometheus](https://prometheus.io/) to collect and visualize queue metrics | ||||
| - [Web UI](#web-ui) to inspect and remote-control queues and tasks | ||||
| @@ -45,13 +44,19 @@ Task queues are used as a mechanism to distribute work across multiple machines. | ||||
|  | ||||
| ## Stability and Compatibility | ||||
|  | ||||
| **Status**: The library is currently undergoing **heavy development** with frequent, breaking API changes. | ||||
| **Status**: The library relatively stable and is currently undergoing **moderate development** with less frequent breaking API changes. | ||||
|  | ||||
| > ☝️ **Important Note**: Current major version is zero (`v0.x.x`) to accommodate rapid development and fast iteration while getting early feedback from users (_feedback on APIs are appreciated!_). The public API could change without a major version update before `v1.0.0` release. | ||||
|  | ||||
| ## Quickstart | ||||
| ### Redis Cluster Compatibility | ||||
|  | ||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). Latest two Go versions are supported (See https://go.dev/dl). | ||||
| Some of the lua scripts in this library may not be compatible with Redis Cluster. | ||||
|  | ||||
| ## Sponsoring | ||||
| If you are using this package in production, **please consider sponsoring the project to show your support!** | ||||
|  | ||||
| ## Quickstart | ||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). The **last 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: | ||||
|  | ||||
| @@ -292,7 +297,7 @@ Asynq ships with a command line tool to inspect the state of queues and tasks. | ||||
| To install the CLI tool, run the following command: | ||||
|  | ||||
| ```sh | ||||
| go install github.com/hibiken/asynq/tools/asynq | ||||
| go install github.com/hibiken/asynq/tools/asynq@latest | ||||
| ``` | ||||
|  | ||||
| Here's an example of running the `asynq dash` command: | ||||
|   | ||||
							
								
								
									
										4
									
								
								asynq.go
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								asynq.go
									
									
									
									
									
								
							| @@ -316,6 +316,9 @@ type RedisFailoverClientOpt struct { | ||||
| 	// https://redis.io/topics/sentinel. | ||||
| 	SentinelAddrs []string | ||||
|  | ||||
| 	// Redis sentinel username. | ||||
| 	SentinelUsername string | ||||
|  | ||||
| 	// Redis sentinel password. | ||||
| 	SentinelPassword string | ||||
|  | ||||
| @@ -364,6 +367,7 @@ func (opt RedisFailoverClientOpt) MakeRedisClient() interface{} { | ||||
| 	return redis.NewFailoverClient(&redis.FailoverOptions{ | ||||
| 		MasterName:       opt.MasterName, | ||||
| 		SentinelAddrs:    opt.SentinelAddrs, | ||||
| 		SentinelUsername: opt.SentinelUsername, | ||||
| 		SentinelPassword: opt.SentinelPassword, | ||||
| 		Username:         opt.Username, | ||||
| 		Password:         opt.Password, | ||||
|   | ||||
| @@ -55,7 +55,7 @@ func BenchmarkEndToEndSimple(b *testing.B) { | ||||
| 		} | ||||
| 		b.StartTimer() // end setup | ||||
|  | ||||
| 		srv.Start(HandlerFunc(handler)) | ||||
| 		_ = srv.Start(HandlerFunc(handler)) | ||||
| 		wg.Wait() | ||||
|  | ||||
| 		b.StopTimer() // begin teardown | ||||
| @@ -117,7 +117,7 @@ func BenchmarkEndToEnd(b *testing.B) { | ||||
| 		} | ||||
| 		b.StartTimer() // end setup | ||||
|  | ||||
| 		srv.Start(HandlerFunc(handler)) | ||||
| 		_ = srv.Start(HandlerFunc(handler)) | ||||
| 		wg.Wait() | ||||
|  | ||||
| 		b.StopTimer() // begin teardown | ||||
| @@ -174,7 +174,7 @@ func BenchmarkEndToEndMultipleQueues(b *testing.B) { | ||||
| 		} | ||||
| 		b.StartTimer() // end setup | ||||
|  | ||||
| 		srv.Start(HandlerFunc(handler)) | ||||
| 		_ = srv.Start(HandlerFunc(handler)) | ||||
| 		wg.Wait() | ||||
|  | ||||
| 		b.StopTimer() // begin teardown | ||||
| @@ -215,7 +215,7 @@ func BenchmarkClientWhileServerRunning(b *testing.B) { | ||||
| 		handler := func(ctx context.Context, t *Task) error { | ||||
| 			return nil | ||||
| 		} | ||||
| 		srv.Start(HandlerFunc(handler)) | ||||
| 		_ = srv.Start(HandlerFunc(handler)) | ||||
|  | ||||
| 		b.StartTimer() // end setup | ||||
|  | ||||
|   | ||||
							
								
								
									
										33
									
								
								client.go
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								client.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // A Client is responsible for scheduling tasks. | ||||
| @@ -25,15 +25,26 @@ import ( | ||||
| // Clients are safe for concurrent use by multiple goroutines. | ||||
| type Client struct { | ||||
| 	broker base.Broker | ||||
| 	// When a Client has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
| } | ||||
|  | ||||
| // NewClient returns a new Client instance given a redis connection option. | ||||
| func NewClient(r RedisConnOpt) *Client { | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		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 | ||||
| @@ -150,9 +161,9 @@ func (t deadlineOption) Value() interface{} { return time.Time(t) } | ||||
| // TTL duration must be greater than or equal to 1 second. | ||||
| // | ||||
| // Uniqueness of a task is based on the following properties: | ||||
| //     - Task Type | ||||
| //     - Task Payload | ||||
| //     - Queue Name | ||||
| //   - Task Type | ||||
| //   - Task Payload | ||||
| //   - Queue Name | ||||
| func Unique(ttl time.Duration) Option { | ||||
| 	return uniqueOption(ttl) | ||||
| } | ||||
| @@ -307,6 +318,9 @@ var ( | ||||
|  | ||||
| // Close closes the connection with redis. | ||||
| func (c *Client) Close() error { | ||||
| 	if c.sharedConnection { | ||||
| 		return fmt.Errorf("redis connection is shared so the Client can't be closed through asynq") | ||||
| 	} | ||||
| 	return c.broker.Close() | ||||
| } | ||||
|  | ||||
| @@ -405,6 +419,11 @@ func (c *Client) EnqueueContext(ctx context.Context, task *Task, opts ...Option) | ||||
| 	return newTaskInfo(msg, state, opt.processAt, nil), nil | ||||
| } | ||||
|  | ||||
| // Ping performs a ping against the redis connection. | ||||
| func (c *Client) Ping() error { | ||||
| 	return c.broker.Ping() | ||||
| } | ||||
|  | ||||
| func (c *Client) enqueue(ctx context.Context, msg *base.TaskMessage, uniqueTTL time.Duration) error { | ||||
| 	if uniqueTTL > 0 { | ||||
| 		return c.broker.EnqueueUnique(ctx, msg, uniqueTTL) | ||||
| @@ -414,7 +433,7 @@ func (c *Client) enqueue(ctx context.Context, msg *base.TaskMessage, uniqueTTL t | ||||
|  | ||||
| func (c *Client) schedule(ctx context.Context, msg *base.TaskMessage, t time.Time, uniqueTTL time.Duration) error { | ||||
| 	if uniqueTTL > 0 { | ||||
| 		ttl := t.Add(uniqueTTL).Sub(time.Now()) | ||||
| 		ttl := time.Until(t.Add(uniqueTTL)) | ||||
| 		return c.broker.ScheduleUnique(ctx, msg, t, ttl) | ||||
| 	} | ||||
| 	return c.broker.Schedule(ctx, msg, t) | ||||
|   | ||||
| @@ -14,6 +14,7 @@ import ( | ||||
| 	"github.com/google/go-cmp/cmp/cmpopts" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	h "github.com/hibiken/asynq/internal/testutil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| @@ -143,11 +144,7 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueue(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| func testClientEnqueue(t *testing.T, client *Client, r redis.UniversalClient) { | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
| 	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) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| @@ -541,11 +556,11 @@ func TestClientEnqueueWithGroupOption(t *testing.T) { | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With Group and ProcessIn options", | ||||
| 			desc: "With Group and ProcessAt options", | ||||
| 			task: task, | ||||
| 			opts: []Option{ | ||||
| 				Group("mygroup"), | ||||
| 				ProcessIn(30 * time.Minute), | ||||
| 				ProcessAt(now.Add(30 * time.Minute)), | ||||
| 			}, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| @@ -1158,7 +1173,7 @@ func TestClientEnqueueUniqueWithProcessAtOption(t *testing.T) { | ||||
| 		} | ||||
|  | ||||
| 		gotTTL := r.TTL(context.Background(), base.UniqueKey(base.DefaultQueueName, tc.task.Type(), tc.task.Payload())).Val() | ||||
| 		wantTTL := tc.at.Add(tc.ttl).Sub(time.Now()) | ||||
| 		wantTTL := time.Until(tc.at.Add(tc.ttl)) | ||||
| 		if !cmp.Equal(wantTTL.Seconds(), gotTTL.Seconds(), cmpopts.EquateApprox(0, 1)) { | ||||
| 			t.Errorf("TTL = %v, want %v", gotTTL, wantTTL) | ||||
| 			continue | ||||
|   | ||||
							
								
								
									
										25
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,17 +1,20 @@ | ||||
| module github.com/hibiken/asynq | ||||
|  | ||||
| go 1.14 | ||||
| go 1.22 | ||||
|  | ||||
| require ( | ||||
| 	github.com/golang/protobuf v1.5.2 | ||||
| 	github.com/google/go-cmp v0.5.6 | ||||
| 	github.com/google/uuid v1.2.0 | ||||
| 	github.com/redis/go-redis/v9 v9.0.3 | ||||
| 	github.com/google/go-cmp v0.6.0 | ||||
| 	github.com/google/uuid v1.6.0 | ||||
| 	github.com/redis/go-redis/v9 v9.7.0 | ||||
| 	github.com/robfig/cron/v3 v3.0.1 | ||||
| 	github.com/spf13/cast v1.3.1 | ||||
| 	go.uber.org/goleak v1.1.12 | ||||
| 	golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e | ||||
| 	golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 | ||||
| 	google.golang.org/protobuf v1.26.0 | ||||
| 	gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect | ||||
| 	github.com/spf13/cast v1.7.0 | ||||
| 	go.uber.org/goleak v1.3.0 | ||||
| 	golang.org/x/sys v0.27.0 | ||||
| 	golang.org/x/time v0.8.0 | ||||
| 	google.golang.org/protobuf v1.35.2 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/cespare/xxhash/v2 v2.2.0 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										104
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										104
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,82 +1,42 @@ | ||||
| github.com/bsm/ginkgo/v2 v2.7.0 h1:ItPMPH90RbmZJt5GtkcNvIRuGEdwlBItdNVoyzaNQao= | ||||
| github.com/bsm/ginkgo/v2 v2.7.0/go.mod h1:AiKlXPm7ItEHNc/2+OkrNG4E0ITzojb9/xWzvQ9XZ9w= | ||||
| github.com/bsm/gomega v1.26.0 h1:LhQm+AFcgV2M0WyKroMASzAzCAJVpAxQXv4SaI9a69Y= | ||||
| github.com/bsm/gomega v1.26.0/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= | ||||
| github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/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/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/uuid v1.2.0 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||
| github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| 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.0.3 h1:+7mmR26M0IvyLxGZUHxu4GiBkJkVDid0Un+j4ScYu4k= | ||||
| github.com/redis/go-redis/v9 v9.0.3/go.mod h1:WqMKv5vnQbRuZstUwxQI195wHy+t4PuXDOjzMvcuQHk= | ||||
| github.com/redis/go-redis/v9 v9.7.0 h1:HhLSs+B6O021gwzl+locl0zEDnyNkxMtf/Z3NNBMa9E= | ||||
| github.com/redis/go-redis/v9 v9.7.0/go.mod h1:f6zhXITC7JUJIlPEiBOTXxJgPLdZcA93GewI7inzyWw= | ||||
| 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/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= | ||||
| github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||
| go.uber.org/goleak v1.1.12 h1:gZAh5/EyT/HQwlpkCy6wTpqfH9H8Lz8zbm3dZh+OyzA= | ||||
| go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de h1:5hukYrvBGR8/eNkX5mdUezrA6JiaEZDtJb9Ei+1LlBs= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.4.2 h1:Gz96sIWK3OalVv/I/qNygP42zyoKp3xptRVCWRFEBvo= | ||||
| golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e h1:fLOSk5Q00efkSvAm+4xcoXD+RRmLmmulPn5I3Y9F2EM= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.1.5 h1:ouewzE6p+/VEB31YYnTbEJdi8pFqKp4P4n85vwo3DHA= | ||||
| golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||||
| github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= | ||||
| github.com/spf13/cast v1.7.0 h1:ntdiHjuueXFgm5nzDRdOS4yfT43P5Fnud6DH50rz/7w= | ||||
| github.com/spf13/cast v1.7.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= | ||||
| github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= | ||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | ||||
| go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||
| go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||
| golang.org/x/sys v0.27.0 h1:wBqf8DvsY9Y/2P8gAfPDEYNuS30J4lPHJxXSb/nJZ+s= | ||||
| golang.org/x/sys v0.27.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/time v0.8.0 h1:9i3RxcPv3PZnitoVGMPDKZSq1xW1gK1Xy3ArNOGZfEg= | ||||
| golang.org/x/time v0.8.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| google.golang.org/protobuf v1.35.2 h1:8Ar7bF+apOIoThw1EdZl0p1oWvMqTHmpA2fRTyZO8io= | ||||
| google.golang.org/protobuf v1.35.2/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
|   | ||||
| @@ -120,7 +120,9 @@ func (h *heartbeater) start(wg *sync.WaitGroup) { | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-h.done: | ||||
| 				h.broker.ClearServerState(h.host, h.pid, h.serverID) | ||||
| 				if err := h.broker.ClearServerState(h.host, h.pid, h.serverID); err != nil { | ||||
| 					h.logger.Errorf("Failed to clear server state: %v", err) | ||||
| 				} | ||||
| 				h.logger.Debug("Heartbeater done") | ||||
| 				timer.Stop() | ||||
| 				return | ||||
|   | ||||
							
								
								
									
										19
									
								
								inspector.go
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								inspector.go
									
									
									
									
									
								
							| @@ -10,16 +10,19 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // Inspector is a client interface to inspect and mutate the state of | ||||
| // queues and tasks. | ||||
| type Inspector struct { | ||||
| 	rdb *rdb.RDB | ||||
| 	// When an Inspector has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
| } | ||||
|  | ||||
| // New returns a new instance of Inspector. | ||||
| @@ -28,13 +31,25 @@ func NewInspector(r RedisConnOpt) *Inspector { | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("inspeq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	inspector := NewInspectorFromRedisClient(c) | ||||
| 	inspector.sharedConnection = false | ||||
| 	return inspector | ||||
| } | ||||
|  | ||||
| // NewInspectorFromRedisClient returns a new instance of Inspector given a redis.UniversalClient | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewInspectorFromRedisClient(c redis.UniversalClient) *Inspector { | ||||
| 	return &Inspector{ | ||||
| 		rdb: rdb.NewRDB(c), | ||||
| 		rdb:              rdb.NewRDB(c), | ||||
| 		sharedConnection: true, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Close closes the connection with redis. | ||||
| func (i *Inspector) Close() error { | ||||
| 	if i.sharedConnection { | ||||
| 		return fmt.Errorf("redis connection is shared so the Inspector can't be closed through asynq") | ||||
| 	} | ||||
| 	return i.rdb.Close() | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -22,11 +22,7 @@ import ( | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| func TestInspectorQueues(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	inspector := NewInspector(getRedisConnOpt(t)) | ||||
|  | ||||
| func testInspectorQueues(t *testing.T, inspector *Inspector, r redis.UniversalClient) { | ||||
| 	tests := []struct { | ||||
| 		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) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| 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) { | ||||
|   | ||||
| @@ -14,16 +14,16 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/golang/protobuf/ptypes" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	pb "github.com/hibiken/asynq/internal/proto" | ||||
| 	"github.com/hibiken/asynq/internal/timeutil" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| 	"google.golang.org/protobuf/types/known/timestamppb" | ||||
| ) | ||||
|  | ||||
| // Version of asynq library and CLI. | ||||
| const Version = "0.24.1" | ||||
| const Version = "0.25.0" | ||||
|  | ||||
| // DefaultQueueName is the queue name used if none are specified by user. | ||||
| const DefaultQueueName = "default" | ||||
| @@ -104,76 +104,76 @@ func ValidateQueueName(qname string) error { | ||||
|  | ||||
| // QueueKeyPrefix returns a prefix for all keys in the given queue. | ||||
| func QueueKeyPrefix(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:", qname) | ||||
| 	return "asynq:{" + qname + "}:" | ||||
| } | ||||
|  | ||||
| // TaskKeyPrefix returns a prefix for task key. | ||||
| func TaskKeyPrefix(qname string) string { | ||||
| 	return fmt.Sprintf("%st:", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "t:" | ||||
| } | ||||
|  | ||||
| // TaskKey returns a redis key for the given task message. | ||||
| func TaskKey(qname, id string) string { | ||||
| 	return fmt.Sprintf("%s%s", TaskKeyPrefix(qname), id) | ||||
| 	return TaskKeyPrefix(qname) + id | ||||
| } | ||||
|  | ||||
| // PendingKey returns a redis key for the given queue name. | ||||
| func PendingKey(qname string) string { | ||||
| 	return fmt.Sprintf("%spending", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "pending" | ||||
| } | ||||
|  | ||||
| // ActiveKey returns a redis key for the active tasks. | ||||
| func ActiveKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sactive", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "active" | ||||
| } | ||||
|  | ||||
| // ScheduledKey returns a redis key for the scheduled tasks. | ||||
| func ScheduledKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sscheduled", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "scheduled" | ||||
| } | ||||
|  | ||||
| // RetryKey returns a redis key for the retry tasks. | ||||
| func RetryKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sretry", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "retry" | ||||
| } | ||||
|  | ||||
| // ArchivedKey returns a redis key for the archived tasks. | ||||
| func ArchivedKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sarchived", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "archived" | ||||
| } | ||||
|  | ||||
| // LeaseKey returns a redis key for the lease. | ||||
| func LeaseKey(qname string) string { | ||||
| 	return fmt.Sprintf("%slease", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "lease" | ||||
| } | ||||
|  | ||||
| func CompletedKey(qname string) string { | ||||
| 	return fmt.Sprintf("%scompleted", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "completed" | ||||
| } | ||||
|  | ||||
| // PausedKey returns a redis key to indicate that the given queue is paused. | ||||
| func PausedKey(qname string) string { | ||||
| 	return fmt.Sprintf("%spaused", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "paused" | ||||
| } | ||||
|  | ||||
| // ProcessedTotalKey returns a redis key for total processed count for the given queue. | ||||
| func ProcessedTotalKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sprocessed", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "processed" | ||||
| } | ||||
|  | ||||
| // FailedTotalKey returns a redis key for total failure count for the given queue. | ||||
| func FailedTotalKey(qname string) string { | ||||
| 	return fmt.Sprintf("%sfailed", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "failed" | ||||
| } | ||||
|  | ||||
| // ProcessedKey returns a redis key for processed count for the given day for the queue. | ||||
| func ProcessedKey(qname string, t time.Time) string { | ||||
| 	return fmt.Sprintf("%sprocessed:%s", QueueKeyPrefix(qname), t.UTC().Format("2006-01-02")) | ||||
| 	return QueueKeyPrefix(qname) + "processed:" + t.UTC().Format("2006-01-02") | ||||
| } | ||||
|  | ||||
| // FailedKey returns a redis key for failure count for the given day for the queue. | ||||
| func FailedKey(qname string, t time.Time) string { | ||||
| 	return fmt.Sprintf("%sfailed:%s", QueueKeyPrefix(qname), t.UTC().Format("2006-01-02")) | ||||
| 	return QueueKeyPrefix(qname) + "failed:" + t.UTC().Format("2006-01-02") | ||||
| } | ||||
|  | ||||
| // ServerInfoKey returns a redis key for process info. | ||||
| @@ -188,47 +188,47 @@ func WorkersKey(hostname string, pid int, serverID string) string { | ||||
|  | ||||
| // SchedulerEntriesKey returns a redis key for the scheduler entries given scheduler ID. | ||||
| func SchedulerEntriesKey(schedulerID string) string { | ||||
| 	return fmt.Sprintf("asynq:schedulers:{%s}", schedulerID) | ||||
| 	return "asynq:schedulers:{" + schedulerID + "}" | ||||
| } | ||||
|  | ||||
| // SchedulerHistoryKey returns a redis key for the scheduler's history for the given entry. | ||||
| func SchedulerHistoryKey(entryID string) string { | ||||
| 	return fmt.Sprintf("asynq:scheduler_history:%s", entryID) | ||||
| 	return "asynq:scheduler_history:" + entryID | ||||
| } | ||||
|  | ||||
| // UniqueKey returns a redis key with the given type, payload, and queue name. | ||||
| func UniqueKey(qname, tasktype string, payload []byte) string { | ||||
| 	if payload == nil { | ||||
| 		return fmt.Sprintf("%sunique:%s:", QueueKeyPrefix(qname), tasktype) | ||||
| 		return QueueKeyPrefix(qname) + "unique:" + tasktype + ":" | ||||
| 	} | ||||
| 	checksum := md5.Sum(payload) | ||||
| 	return fmt.Sprintf("%sunique:%s:%s", QueueKeyPrefix(qname), tasktype, hex.EncodeToString(checksum[:])) | ||||
| 	return QueueKeyPrefix(qname) + "unique:" + tasktype + ":" + hex.EncodeToString(checksum[:]) | ||||
| } | ||||
|  | ||||
| // GroupKeyPrefix returns a prefix for group key. | ||||
| func GroupKeyPrefix(qname string) string { | ||||
| 	return fmt.Sprintf("%sg:", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "g:" | ||||
| } | ||||
|  | ||||
| // GroupKey returns a redis key used to group tasks belong in the same group. | ||||
| func GroupKey(qname, gkey string) string { | ||||
| 	return fmt.Sprintf("%s%s", GroupKeyPrefix(qname), gkey) | ||||
| 	return GroupKeyPrefix(qname) + gkey | ||||
| } | ||||
|  | ||||
| // AggregationSetKey returns a redis key used for an aggregation set. | ||||
| func AggregationSetKey(qname, gname, setID string) string { | ||||
| 	return fmt.Sprintf("%s:%s", GroupKey(qname, gname), setID) | ||||
| 	return GroupKey(qname, gname) + ":" + setID | ||||
| } | ||||
|  | ||||
| // AllGroups return a redis key used to store all group keys used in a given queue. | ||||
| func AllGroups(qname string) string { | ||||
| 	return fmt.Sprintf("%sgroups", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "groups" | ||||
| } | ||||
|  | ||||
| // AllAggregationSets returns a redis key used to store all aggregation sets (set of tasks staged to be aggregated) | ||||
| // in a given queue. | ||||
| func AllAggregationSets(qname string) string { | ||||
| 	return fmt.Sprintf("%saggregation_sets", QueueKeyPrefix(qname)) | ||||
| 	return QueueKeyPrefix(qname) + "aggregation_sets" | ||||
| } | ||||
|  | ||||
| // TaskMessage is the internal representation of a task with additional metadata fields. | ||||
| @@ -375,14 +375,12 @@ func EncodeServerInfo(info *ServerInfo) ([]byte, error) { | ||||
| 	if info == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil server info") | ||||
| 	} | ||||
| 	queues := make(map[string]int32) | ||||
| 	queues := make(map[string]int32, len(info.Queues)) | ||||
| 	for q, p := range info.Queues { | ||||
| 		queues[q] = int32(p) | ||||
| 	} | ||||
| 	started, err := ptypes.TimestampProto(info.Started) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	started := timestamppb.New(info.Started) | ||||
|  | ||||
| 	return proto.Marshal(&pb.ServerInfo{ | ||||
| 		Host:              info.Host, | ||||
| 		Pid:               int32(info.PID), | ||||
| @@ -402,14 +400,12 @@ func DecodeServerInfo(b []byte) (*ServerInfo, error) { | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	queues := make(map[string]int) | ||||
| 	queues := make(map[string]int, len(pbmsg.GetQueues())) | ||||
| 	for q, p := range pbmsg.GetQueues() { | ||||
| 		queues[q] = int(p) | ||||
| 	} | ||||
| 	startTime, err := ptypes.Timestamp(pbmsg.GetStartTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	startTime := pbmsg.GetStartTime() | ||||
|  | ||||
| 	return &ServerInfo{ | ||||
| 		Host:              pbmsg.GetHost(), | ||||
| 		PID:               int(pbmsg.GetPid()), | ||||
| @@ -418,7 +414,7 @@ func DecodeServerInfo(b []byte) (*ServerInfo, error) { | ||||
| 		Queues:            queues, | ||||
| 		StrictPriority:    pbmsg.GetStrictPriority(), | ||||
| 		Status:            pbmsg.GetStatus(), | ||||
| 		Started:           startTime, | ||||
| 		Started:           startTime.AsTime(), | ||||
| 		ActiveWorkerCount: int(pbmsg.GetActiveWorkerCount()), | ||||
| 	}, nil | ||||
| } | ||||
| @@ -441,14 +437,9 @@ func EncodeWorkerInfo(info *WorkerInfo) ([]byte, error) { | ||||
| 	if info == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil worker info") | ||||
| 	} | ||||
| 	startTime, err := ptypes.TimestampProto(info.Started) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	deadline, err := ptypes.TimestampProto(info.Deadline) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	startTime := timestamppb.New(info.Started) | ||||
| 	deadline := timestamppb.New(info.Deadline) | ||||
|  | ||||
| 	return proto.Marshal(&pb.WorkerInfo{ | ||||
| 		Host:        info.Host, | ||||
| 		Pid:         int32(info.PID), | ||||
| @@ -468,14 +459,9 @@ func DecodeWorkerInfo(b []byte) (*WorkerInfo, error) { | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	startTime, err := ptypes.Timestamp(pbmsg.GetStartTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	deadline, err := ptypes.Timestamp(pbmsg.GetDeadline()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	startTime := pbmsg.GetStartTime() | ||||
| 	deadline := pbmsg.GetDeadline() | ||||
|  | ||||
| 	return &WorkerInfo{ | ||||
| 		Host:     pbmsg.GetHost(), | ||||
| 		PID:      int(pbmsg.GetPid()), | ||||
| @@ -484,8 +470,8 @@ func DecodeWorkerInfo(b []byte) (*WorkerInfo, error) { | ||||
| 		Type:     pbmsg.GetTaskType(), | ||||
| 		Payload:  pbmsg.GetTaskPayload(), | ||||
| 		Queue:    pbmsg.GetQueue(), | ||||
| 		Started:  startTime, | ||||
| 		Deadline: deadline, | ||||
| 		Started:  startTime.AsTime(), | ||||
| 		Deadline: deadline.AsTime(), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -519,14 +505,9 @@ func EncodeSchedulerEntry(entry *SchedulerEntry) ([]byte, error) { | ||||
| 	if entry == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil scheduler entry") | ||||
| 	} | ||||
| 	next, err := ptypes.TimestampProto(entry.Next) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	prev, err := ptypes.TimestampProto(entry.Prev) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	next := timestamppb.New(entry.Next) | ||||
| 	prev := timestamppb.New(entry.Prev) | ||||
|  | ||||
| 	return proto.Marshal(&pb.SchedulerEntry{ | ||||
| 		Id:              entry.ID, | ||||
| 		Spec:            entry.Spec, | ||||
| @@ -544,22 +525,17 @@ func DecodeSchedulerEntry(b []byte) (*SchedulerEntry, error) { | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	next, err := ptypes.Timestamp(pbmsg.GetNextEnqueueTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	prev, err := ptypes.Timestamp(pbmsg.GetPrevEnqueueTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	next := pbmsg.GetNextEnqueueTime() | ||||
| 	prev := pbmsg.GetPrevEnqueueTime() | ||||
|  | ||||
| 	return &SchedulerEntry{ | ||||
| 		ID:      pbmsg.GetId(), | ||||
| 		Spec:    pbmsg.GetSpec(), | ||||
| 		Type:    pbmsg.GetTaskType(), | ||||
| 		Payload: pbmsg.GetTaskPayload(), | ||||
| 		Opts:    pbmsg.GetEnqueueOptions(), | ||||
| 		Next:    next, | ||||
| 		Prev:    prev, | ||||
| 		Next:    next.AsTime(), | ||||
| 		Prev:    prev.AsTime(), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -578,10 +554,7 @@ func EncodeSchedulerEnqueueEvent(event *SchedulerEnqueueEvent) ([]byte, error) { | ||||
| 	if event == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil enqueue event") | ||||
| 	} | ||||
| 	enqueuedAt, err := ptypes.TimestampProto(event.EnqueuedAt) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	enqueuedAt := timestamppb.New(event.EnqueuedAt) | ||||
| 	return proto.Marshal(&pb.SchedulerEnqueueEvent{ | ||||
| 		TaskId:      event.TaskID, | ||||
| 		EnqueueTime: enqueuedAt, | ||||
| @@ -595,13 +568,10 @@ func DecodeSchedulerEnqueueEvent(b []byte) (*SchedulerEnqueueEvent, error) { | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	enqueuedAt, err := ptypes.Timestamp(pbmsg.GetEnqueueTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	enqueuedAt := pbmsg.GetEnqueueTime() | ||||
| 	return &SchedulerEnqueueEvent{ | ||||
| 		TaskID:     pbmsg.GetTaskId(), | ||||
| 		EnqueuedAt: enqueuedAt, | ||||
| 		EnqueuedAt: enqueuedAt.AsTime(), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -737,7 +707,7 @@ type Broker interface { | ||||
| 	ReclaimStaleAggregationSets(qname string) error | ||||
|  | ||||
| 	// Task retention related method | ||||
| 	DeleteExpiredCompletedTasks(qname string) error | ||||
| 	DeleteExpiredCompletedTasks(qname string, batchSize int) error | ||||
|  | ||||
| 	// Lease related methods | ||||
| 	ListLeaseExpired(cutoff time.Time, qnames ...string) ([]*TaskMessage, error) | ||||
|   | ||||
| @@ -256,6 +256,21 @@ func IsRedisCommandError(err error) bool { | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // PanicError defines an error when occurred a panic error. | ||||
| type PanicError struct { | ||||
| 	ErrMsg string | ||||
| } | ||||
|  | ||||
| func (e *PanicError) Error() string { | ||||
| 	return fmt.Sprintf("panic error cause by: %s", e.ErrMsg) | ||||
| } | ||||
|  | ||||
| // IsPanicError reports whether any error in err's chain is of type PanicError. | ||||
| func IsPanicError(err error) bool { | ||||
| 	var target *PanicError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| /************************************************* | ||||
|     Standard Library errors package functions | ||||
| *************************************************/ | ||||
|   | ||||
| @@ -131,6 +131,12 @@ func TestErrorPredicates(t *testing.T) { | ||||
| 			err:  E(Op("rdb.ArchiveTask"), NotFound, &QueueNotFoundError{Queue: "default"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "IsPanicError should detect presence of PanicError in err's chain", | ||||
| 			fn:   IsPanicError, | ||||
| 			err:  E(Op("unknown"), Unknown, &PanicError{ErrMsg: "Something went wrong"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
|   | ||||
| @@ -4,14 +4,13 @@ | ||||
|  | ||||
| // Code generated by protoc-gen-go. DO NOT EDIT. | ||||
| // versions: | ||||
| // 	protoc-gen-go v1.25.0 | ||||
| // 	protoc        v3.17.3 | ||||
| // 	protoc-gen-go v1.34.2 | ||||
| // 	protoc        v3.19.6 | ||||
| // source: asynq.proto | ||||
|  | ||||
| package proto | ||||
|  | ||||
| import ( | ||||
| 	proto "github.com/golang/protobuf/proto" | ||||
| 	protoreflect "google.golang.org/protobuf/reflect/protoreflect" | ||||
| 	protoimpl "google.golang.org/protobuf/runtime/protoimpl" | ||||
| 	timestamppb "google.golang.org/protobuf/types/known/timestamppb" | ||||
| @@ -26,10 +25,6 @@ const ( | ||||
| 	_ = protoimpl.EnforceVersion(protoimpl.MaxVersion - 20) | ||||
| ) | ||||
|  | ||||
| // This is a compile-time assertion that a sufficiently up-to-date version | ||||
| // of the legacy proto package is being used. | ||||
| const _ = proto.ProtoPackageIsVersion4 | ||||
|  | ||||
| // TaskMessage is the internal representation of a task with additional | ||||
| // metadata fields. | ||||
| // Next ID: 15 | ||||
| @@ -739,7 +734,7 @@ func file_asynq_proto_rawDescGZIP() []byte { | ||||
| } | ||||
|  | ||||
| var file_asynq_proto_msgTypes = make([]protoimpl.MessageInfo, 6) | ||||
| var file_asynq_proto_goTypes = []interface{}{ | ||||
| var file_asynq_proto_goTypes = []any{ | ||||
| 	(*TaskMessage)(nil),           // 0: asynq.TaskMessage | ||||
| 	(*ServerInfo)(nil),            // 1: asynq.ServerInfo | ||||
| 	(*WorkerInfo)(nil),            // 2: asynq.WorkerInfo | ||||
| @@ -769,7 +764,7 @@ func file_asynq_proto_init() { | ||||
| 		return | ||||
| 	} | ||||
| 	if !protoimpl.UnsafeEnabled { | ||||
| 		file_asynq_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_asynq_proto_msgTypes[0].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*TaskMessage); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -781,7 +776,7 @@ func file_asynq_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_asynq_proto_msgTypes[1].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*ServerInfo); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -793,7 +788,7 @@ func file_asynq_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_asynq_proto_msgTypes[2].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*WorkerInfo); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -805,7 +800,7 @@ func file_asynq_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_asynq_proto_msgTypes[3].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*SchedulerEntry); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| @@ -817,7 +812,7 @@ func file_asynq_proto_init() { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { | ||||
| 		file_asynq_proto_msgTypes[4].Exporter = func(v any, i int) any { | ||||
| 			switch v := v.(*SchedulerEnqueueEvent); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
|   | ||||
| @@ -10,9 +10,9 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/spf13/cast" | ||||
| ) | ||||
|  | ||||
| @@ -343,7 +343,7 @@ func (r *RDB) memoryUsage(qname string) (int64, error) { | ||||
| 	} | ||||
| 	usg, err := cast.ToInt64E(res) | ||||
| 	if err != nil { | ||||
| 		return 0, errors.E(op, errors.Internal, fmt.Sprintf("could not cast script return value to int64")) | ||||
| 		return 0, errors.E(op, errors.Internal, "could not cast script return value to int64") | ||||
| 	} | ||||
| 	return usg, nil | ||||
| } | ||||
| @@ -1832,6 +1832,7 @@ func (r *RDB) RemoveQueue(qname string, force bool) error { | ||||
| 		if err := r.client.SRem(context.Background(), base.AllQueues, qname).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, err) | ||||
| 		} | ||||
| 		r.queuesPublished.Delete(qname) | ||||
| 		return nil | ||||
| 	case -1: | ||||
| 		return errors.E(op, errors.NotFound, &errors.QueueNotEmptyError{Queue: qname}) | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| @@ -26,8 +27,9 @@ const LeaseDuration = 30 * time.Second | ||||
|  | ||||
| // RDB is a client interface to query and mutate task queues. | ||||
| type RDB struct { | ||||
| 	client redis.UniversalClient | ||||
| 	clock  timeutil.Clock | ||||
| 	client          redis.UniversalClient | ||||
| 	clock           timeutil.Clock | ||||
| 	queuesPublished sync.Map | ||||
| } | ||||
|  | ||||
| // NewRDB returns a new instance of RDB. | ||||
| @@ -112,8 +114,11 @@ func (r *RDB) Enqueue(ctx context.Context, msg *base.TaskMessage) error { | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Unknown, fmt.Sprintf("cannot encode message: %v", err)) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		base.TaskKey(msg.Queue, msg.ID), | ||||
| @@ -174,8 +179,11 @@ func (r *RDB) EnqueueUnique(ctx context.Context, msg *base.TaskMessage, ttl time | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Internal, "cannot encode task message: %v", err) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		msg.UniqueKey, | ||||
| @@ -529,8 +537,11 @@ func (r *RDB) AddToGroup(ctx context.Context, msg *base.TaskMessage, groupKey st | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Unknown, fmt.Sprintf("cannot encode message: %v", err)) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		base.TaskKey(msg.Queue, msg.ID), | ||||
| @@ -591,8 +602,11 @@ func (r *RDB) AddToGroupUnique(ctx context.Context, msg *base.TaskMessage, group | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Unknown, fmt.Sprintf("cannot encode message: %v", err)) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		base.TaskKey(msg.Queue, msg.ID), | ||||
| @@ -648,8 +662,11 @@ func (r *RDB) Schedule(ctx context.Context, msg *base.TaskMessage, processAt tim | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Unknown, fmt.Sprintf("cannot encode message: %v", err)) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		base.TaskKey(msg.Queue, msg.ID), | ||||
| @@ -707,8 +724,11 @@ func (r *RDB) ScheduleUnique(ctx context.Context, msg *base.TaskMessage, process | ||||
| 	if err != nil { | ||||
| 		return errors.E(op, errors.Internal, fmt.Sprintf("cannot encode task message: %v", err)) | ||||
| 	} | ||||
| 	if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 	if _, found := r.queuesPublished.Load(msg.Queue); !found { | ||||
| 		if err := r.client.SAdd(ctx, base.AllQueues, msg.Queue).Err(); err != nil { | ||||
| 			return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "sadd", Err: err}) | ||||
| 		} | ||||
| 		r.queuesPublished.Store(msg.Queue, true) | ||||
| 	} | ||||
| 	keys := []string{ | ||||
| 		msg.UniqueKey, | ||||
| @@ -829,6 +849,7 @@ const ( | ||||
| // KEYS[6] -> asynq:{<qname>}:failed:<yyyy-mm-dd> | ||||
| // KEYS[7] -> asynq:{<qname>}:processed | ||||
| // KEYS[8] -> asynq:{<qname>}:failed | ||||
| // KEYS[9] -> asynq:{<qname>}:t: | ||||
| // ------- | ||||
| // ARGV[1] -> task ID | ||||
| // ARGV[2] -> updated base.TaskMessage value | ||||
| @@ -845,8 +866,22 @@ if redis.call("ZREM", KEYS[3], ARGV[1]) == 0 then | ||||
|   return redis.error_reply("NOT FOUND") | ||||
| end | ||||
| redis.call("ZADD", KEYS[4], ARGV[3], ARGV[1]) | ||||
| redis.call("ZREMRANGEBYSCORE", KEYS[4], "-inf", ARGV[4]) | ||||
| redis.call("ZREMRANGEBYRANK", KEYS[4], 0, -ARGV[5]) | ||||
| local old = redis.call("ZRANGE", KEYS[4], "-inf", ARGV[4], "BYSCORE") | ||||
| if #old > 0 then | ||||
| 	for _, id in ipairs(old) do | ||||
| 		redis.call("DEL", KEYS[9] .. id) | ||||
| 	end | ||||
| 	redis.call("ZREM", KEYS[4], unpack(old)) | ||||
| end | ||||
|  | ||||
| local extra = redis.call("ZRANGE", KEYS[4], 0, -ARGV[5]) | ||||
| if #extra > 0 then | ||||
| 	for _, id in ipairs(extra) do | ||||
| 		redis.call("DEL", KEYS[9] .. id) | ||||
| 	end | ||||
| 	redis.call("ZREM", KEYS[4], unpack(extra)) | ||||
| end | ||||
|  | ||||
| redis.call("HSET", KEYS[1], "msg", ARGV[2], "state", "archived") | ||||
| local n = redis.call("INCR", KEYS[5]) | ||||
| if tonumber(n) == 1 then | ||||
| @@ -889,6 +924,7 @@ func (r *RDB) Archive(ctx context.Context, msg *base.TaskMessage, errMsg string) | ||||
| 		base.FailedKey(msg.Queue, now), | ||||
| 		base.ProcessedTotalKey(msg.Queue), | ||||
| 		base.FailedTotalKey(msg.Queue), | ||||
| 		base.TaskKeyPrefix(msg.Queue), | ||||
| 	} | ||||
| 	argv := []interface{}{ | ||||
| 		msg.ID, | ||||
| @@ -1217,7 +1253,7 @@ redis.call("ZREMRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | ||||
| 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. | ||||
| func (r *RDB) ReclaimStaleAggregationSets(qname string) error { | ||||
| 	var op errors.Op = "RDB.ReclaimStaleAggregationSets" | ||||
| @@ -1241,9 +1277,7 @@ return table.getn(ids)`) | ||||
|  | ||||
| // DeleteExpiredCompletedTasks checks for any expired tasks in the given queue's completed set, | ||||
| // and delete all expired tasks. | ||||
| func (r *RDB) DeleteExpiredCompletedTasks(qname string) error { | ||||
| 	// Note: Do this operation in fix batches to prevent long running script. | ||||
| 	const batchSize = 100 | ||||
| func (r *RDB) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||
| 	for { | ||||
| 		n, err := r.deleteExpiredCompletedTasks(qname, batchSize) | ||||
| 		if err != nil { | ||||
| @@ -1284,7 +1318,10 @@ local res = {} | ||||
| local ids = redis.call("ZRANGEBYSCORE", KEYS[1], "-inf", ARGV[1]) | ||||
| for _, id in ipairs(ids) do | ||||
| 	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 | ||||
| return res | ||||
| `) | ||||
| @@ -1400,7 +1437,7 @@ func (r *RDB) ClearServerState(host string, pid int, serverID string) error { | ||||
|  | ||||
| // KEYS[1]  -> asynq:schedulers:{<schedulerID>} | ||||
| // ARGV[1]  -> TTL in seconds | ||||
| // ARGV[2:] -> schedler entries | ||||
| // ARGV[2:] -> scheduler entries | ||||
| var writeSchedulerEntriesCmd = redis.NewScript(` | ||||
| redis.call("DEL", KEYS[1]) | ||||
| for i = 2, #ARGV do | ||||
| @@ -1431,10 +1468,10 @@ func (r *RDB) WriteSchedulerEntries(schedulerID string, entries []*base.Schedule | ||||
| } | ||||
|  | ||||
| // ClearSchedulerEntries deletes scheduler entries data from redis. | ||||
| func (r *RDB) ClearSchedulerEntries(scheduelrID string) error { | ||||
| func (r *RDB) ClearSchedulerEntries(schedulerID string) error { | ||||
| 	var op errors.Op = "rdb.ClearSchedulerEntries" | ||||
| 	ctx := context.Background() | ||||
| 	key := base.SchedulerEntriesKey(scheduelrID) | ||||
| 	key := base.SchedulerEntriesKey(schedulerID) | ||||
| 	if err := r.client.ZRem(ctx, base.AllSchedulers, key).Err(); err != nil { | ||||
| 		return errors.E(op, errors.Unknown, &errors.RedisCommandError{Command: "zrem", Err: err}) | ||||
| 	} | ||||
|   | ||||
| @@ -160,6 +160,59 @@ func TestEnqueueTaskIdConflictError(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEnqueueQueueCache(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	t1 := h.NewTaskMessageWithQueue("sync1", nil, "low") | ||||
|  | ||||
| 	enqueueTime := time.Now() | ||||
| 	clock := timeutil.NewSimulatedClock(enqueueTime) | ||||
| 	r.SetClock(clock) | ||||
|  | ||||
| 	err := r.Enqueue(context.Background(), t1) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("(*RDB).Enqueue(msg) = %v, want nil", err) | ||||
| 	} | ||||
|  | ||||
| 	// Check queue is in the AllQueues set. | ||||
| 	if !r.client.SIsMember(context.Background(), base.AllQueues, t1.Queue).Val() { | ||||
| 		t.Fatalf("%q is not a member of SET %q", t1.Queue, base.AllQueues) | ||||
| 	} | ||||
|  | ||||
| 	if _, ok := r.queuesPublished.Load(t1.Queue); !ok { | ||||
| 		t.Fatalf("%q is not cached in queuesPublished", t1.Queue) | ||||
| 	} | ||||
|  | ||||
| 	t.Run("remove-queue", func(t *testing.T) { | ||||
| 		err := r.RemoveQueue(t1.Queue, true) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("(*RDB).RemoveQueue(%q, %t) = %v, want nil", t1.Queue, true, err) | ||||
| 		} | ||||
|  | ||||
| 		if _, ok := r.queuesPublished.Load(t1.Queue); ok { | ||||
| 			t.Fatalf("%q is still cached in queuesPublished", t1.Queue) | ||||
| 		} | ||||
|  | ||||
| 		if r.client.SIsMember(context.Background(), base.AllQueues, t1.Queue).Val() { | ||||
| 			t.Fatalf("%q is a member of SET %q", t1.Queue, base.AllQueues) | ||||
| 		} | ||||
|  | ||||
| 		err = r.Enqueue(context.Background(), t1) | ||||
| 		if err != nil { | ||||
| 			t.Fatalf("(*RDB).Enqueue(msg) = %v, want nil", err) | ||||
| 		} | ||||
|  | ||||
| 		// Check queue is in the AllQueues set. | ||||
| 		if !r.client.SIsMember(context.Background(), base.AllQueues, t1.Queue).Val() { | ||||
| 			t.Fatalf("%q is not a member of SET %q", t1.Queue, base.AllQueues) | ||||
| 		} | ||||
|  | ||||
| 		if _, ok := r.queuesPublished.Load(t1.Queue); !ok { | ||||
| 			t.Fatalf("%q is not cached in queuesPublished", t1.Queue) | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestEnqueueUnique(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| @@ -2002,7 +2055,6 @@ func TestArchive(t *testing.T) { | ||||
| 	} | ||||
| 	errMsg := "SMTP server not responding" | ||||
|  | ||||
| 	// TODO(hibiken): add test cases for trimming | ||||
| 	tests := []struct { | ||||
| 		active       map[string][]*base.TaskMessage | ||||
| 		lease        map[string][]base.Z | ||||
| @@ -2171,6 +2223,163 @@ func TestArchive(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestArchiveTrim(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	now := time.Now() | ||||
| 	r.SetClock(timeutil.NewSimulatedClock(now)) | ||||
|  | ||||
| 	t1 := &base.TaskMessage{ | ||||
| 		ID:      uuid.NewString(), | ||||
| 		Type:    "send_email", | ||||
| 		Payload: nil, | ||||
| 		Queue:   "default", | ||||
| 		Retry:   25, | ||||
| 		Retried: 25, | ||||
| 		Timeout: 1800, | ||||
| 	} | ||||
| 	t2 := &base.TaskMessage{ | ||||
| 		ID:      uuid.NewString(), | ||||
| 		Type:    "reindex", | ||||
| 		Payload: nil, | ||||
| 		Queue:   "default", | ||||
| 		Retry:   25, | ||||
| 		Retried: 0, | ||||
| 		Timeout: 3000, | ||||
| 	} | ||||
| 	errMsg := "SMTP server not responding" | ||||
|  | ||||
| 	maxArchiveSet := make([]base.Z, 0) | ||||
| 	for i := 0; i < maxArchiveSize-1; i++ { | ||||
| 		maxArchiveSet = append(maxArchiveSet, base.Z{Message: &base.TaskMessage{ | ||||
| 			ID:      uuid.NewString(), | ||||
| 			Type:    "generate_csv", | ||||
| 			Payload: nil, | ||||
| 			Queue:   "default", | ||||
| 			Retry:   25, | ||||
| 			Retried: 0, | ||||
| 			Timeout: 60, | ||||
| 		}, Score: now.Add(-time.Hour + -time.Second*time.Duration(i)).Unix()}) | ||||
| 	} | ||||
|  | ||||
| 	wantMaxArchiveSet := make([]base.Z, 0) | ||||
| 	// newly archived task should be at the front | ||||
| 	wantMaxArchiveSet = append(wantMaxArchiveSet, base.Z{Message: h.TaskMessageWithError(*t1, errMsg, now), Score: now.Unix()}) | ||||
| 	// oldest task should be dropped from the set | ||||
| 	wantMaxArchiveSet = append(wantMaxArchiveSet, maxArchiveSet[:len(maxArchiveSet)-1]...) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		toArchive    map[string][]*base.TaskMessage | ||||
| 		lease        map[string][]base.Z | ||||
| 		archived     map[string][]base.Z | ||||
| 		wantArchived map[string][]base.Z | ||||
| 	}{ | ||||
| 		{ // simple, 1 to be archived, 1 already archived, both are in the archive set | ||||
| 			toArchive: map[string][]*base.TaskMessage{ | ||||
| 				"default": {t1}, | ||||
| 			}, | ||||
| 			lease: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: t1, Score: now.Add(10 * time.Second).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			archived: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: t2, Score: now.Add(-time.Hour).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: h.TaskMessageWithError(*t1, errMsg, now), Score: now.Unix()}, | ||||
| 					{Message: t2, Score: now.Add(-time.Hour).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ // 1 to be archived, 1 already archived but past expiry, only the newly archived task should be left | ||||
| 			toArchive: map[string][]*base.TaskMessage{ | ||||
| 				"default": {t1}, | ||||
| 			}, | ||||
| 			lease: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: t1, Score: now.Add(10 * time.Second).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			archived: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: t2, Score: now.Add(-time.Hour * 24 * (archivedExpirationInDays + 1)).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: h.TaskMessageWithError(*t1, errMsg, now), Score: now.Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ // 1 to be archived, maxArchiveSize in archive set, archive set should be trimmed back to maxArchiveSize and newly archived task should be in the set | ||||
| 			toArchive: map[string][]*base.TaskMessage{ | ||||
| 				"default": {t1}, | ||||
| 			}, | ||||
| 			lease: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: t1, Score: now.Add(10 * time.Second).Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			archived: map[string][]base.Z{ | ||||
| 				"default": maxArchiveSet, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]base.Z{ | ||||
| 				"default": wantMaxArchiveSet, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r.client) // clean up db before each test case | ||||
| 		h.SeedAllActiveQueues(t, r.client, tc.toArchive) | ||||
| 		h.SeedAllLease(t, r.client, tc.lease) | ||||
| 		h.SeedAllArchivedQueues(t, r.client, tc.archived) | ||||
|  | ||||
| 		for _, tasks := range tc.toArchive { | ||||
| 			for _, target := range tasks { | ||||
| 				err := r.Archive(context.Background(), target, errMsg) | ||||
| 				if err != nil { | ||||
| 					t.Errorf("(*RDB).Archive(%v, %v) = %v, want nil", target, errMsg, err) | ||||
| 					continue | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		for queue, want := range tc.wantArchived { | ||||
| 			gotArchived := h.GetArchivedEntries(t, r.client, queue) | ||||
|  | ||||
| 			if diff := cmp.Diff(want, gotArchived, h.SortZSetEntryOpt, zScoreCmpOpt, timeCmpOpt); diff != "" { | ||||
| 				t.Errorf("mismatch found in %q after calling (*RDB).Archive: (-want, +got):\n%s", base.ArchivedKey(queue), diff) | ||||
| 			} | ||||
|  | ||||
| 			// check that only keys present in the archived set are in rdb | ||||
| 			vals := r.client.Keys(context.Background(), base.TaskKeyPrefix(queue)+"*").Val() | ||||
| 			if len(vals) != len(gotArchived) { | ||||
| 				t.Errorf("len of keys = %v, want %v", len(vals), len(gotArchived)) | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			for _, val := range vals { | ||||
| 				found := false | ||||
| 				for _, entry := range gotArchived { | ||||
| 					if strings.Contains(val, entry.Message.ID) { | ||||
| 						found = true | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				if !found { | ||||
| 					t.Errorf("key %v not found in archived set (it was orphaned by the archive trim)", val) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestForwardIfReadyWithGroup(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| @@ -2542,8 +2751,8 @@ func TestDeleteExpiredCompletedTasks(t *testing.T) { | ||||
| 		h.FlushDB(t, r.client) | ||||
| 		h.SeedAllCompletedQueues(t, r.client, tc.completed) | ||||
|  | ||||
| 		if err := r.DeleteExpiredCompletedTasks(tc.qname); err != nil { | ||||
| 			t.Errorf("DeleteExpiredCompletedTasks(%q) failed: %v", tc.qname, err) | ||||
| 		if err := r.DeleteExpiredCompletedTasks(tc.qname, 100); err != nil { | ||||
| 			t.Errorf("DeleteExpiredCompletedTasks(%q, 100) failed: %v", tc.qname, err) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| @@ -3050,7 +3259,7 @@ func TestCancelationPubSub(t *testing.T) { | ||||
| 	publish := []string{"one", "two", "three"} | ||||
|  | ||||
| 	for _, msg := range publish { | ||||
| 		r.PublishCancelation(msg) | ||||
| 		_ = r.PublishCancelation(msg) | ||||
| 	} | ||||
|  | ||||
| 	// allow for message to reach subscribers. | ||||
|   | ||||
| @@ -11,8 +11,8 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| var errRedisDown = errors.New("testutil: redis is down") | ||||
| @@ -145,13 +145,13 @@ func (tb *TestBroker) ForwardIfReady(qnames ...string) error { | ||||
| 	return tb.real.ForwardIfReady(qnames...) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string) error { | ||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string, batchSize int) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		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) { | ||||
|   | ||||
							
								
								
									
										15
									
								
								janitor.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								janitor.go
									
									
									
									
									
								
							| @@ -27,13 +27,17 @@ type janitor struct { | ||||
|  | ||||
| 	// average interval between checks. | ||||
| 	avgInterval time.Duration | ||||
|  | ||||
| 	// number of tasks to be deleted when janitor runs to delete the expired completed tasks. | ||||
| 	batchSize int | ||||
| } | ||||
|  | ||||
| type janitorParams struct { | ||||
| 	logger   *log.Logger | ||||
| 	broker   base.Broker | ||||
| 	queues   []string | ||||
| 	interval time.Duration | ||||
| 	logger    *log.Logger | ||||
| 	broker    base.Broker | ||||
| 	queues    []string | ||||
| 	interval  time.Duration | ||||
| 	batchSize int | ||||
| } | ||||
|  | ||||
| func newJanitor(params janitorParams) *janitor { | ||||
| @@ -43,6 +47,7 @@ func newJanitor(params janitorParams) *janitor { | ||||
| 		done:        make(chan struct{}), | ||||
| 		queues:      params.queues, | ||||
| 		avgInterval: params.interval, | ||||
| 		batchSize:   params.batchSize, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -73,7 +78,7 @@ func (j *janitor) start(wg *sync.WaitGroup) { | ||||
|  | ||||
| func (j *janitor) exec() { | ||||
| 	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", | ||||
| 				qname, err) | ||||
| 		} | ||||
|   | ||||
| @@ -26,11 +26,13 @@ func TestJanitor(t *testing.T) { | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
| 	const interval = 1 * time.Second | ||||
| 	const batchSize = 100 | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:   testLogger, | ||||
| 		broker:   rdbClient, | ||||
| 		queues:   []string{"default", "custom"}, | ||||
| 		interval: interval, | ||||
| 		logger:    testLogger, | ||||
| 		broker:    rdbClient, | ||||
| 		queues:    []string{"default", "custom"}, | ||||
| 		interval:  interval, | ||||
| 		batchSize: batchSize, | ||||
| 	}) | ||||
|  | ||||
| 	now := time.Now() | ||||
|   | ||||
| @@ -7,10 +7,11 @@ package asynq | ||||
| import ( | ||||
| 	"crypto/sha256" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"sort" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // PeriodicTaskManager manages scheduling of periodic tasks. | ||||
| @@ -28,9 +29,12 @@ type PeriodicTaskManagerOpts struct { | ||||
| 	// Required: must be non nil | ||||
| 	PeriodicTaskConfigProvider PeriodicTaskConfigProvider | ||||
|  | ||||
| 	// Required: must be non nil | ||||
| 	// Optional: if RedisUniversalClient is nil must be non nil | ||||
| 	RedisConnOpt RedisConnOpt | ||||
|  | ||||
| 	// Optional: if RedisUniversalClient is non nil, RedisConnOpt is ignored. | ||||
| 	RedisUniversalClient redis.UniversalClient | ||||
|  | ||||
| 	// Optional: scheduler options | ||||
| 	*SchedulerOpts | ||||
|  | ||||
| @@ -46,10 +50,16 @@ func NewPeriodicTaskManager(opts PeriodicTaskManagerOpts) (*PeriodicTaskManager, | ||||
| 	if opts.PeriodicTaskConfigProvider == nil { | ||||
| 		return nil, fmt.Errorf("PeriodicTaskConfigProvider cannot be nil") | ||||
| 	} | ||||
| 	if opts.RedisConnOpt == nil { | ||||
| 		return nil, fmt.Errorf("RedisConnOpt cannot be nil") | ||||
| 	if opts.RedisConnOpt == nil && opts.RedisUniversalClient == nil { | ||||
| 		return nil, fmt.Errorf("RedisConnOpt/RedisUniversalClient cannot be nil") | ||||
| 	} | ||||
| 	scheduler := NewScheduler(opts.RedisConnOpt, opts.SchedulerOpts) | ||||
| 	var scheduler *Scheduler | ||||
| 	if opts.RedisUniversalClient != nil { | ||||
| 		scheduler = NewSchedulerFromRedisClient(opts.RedisUniversalClient, opts.SchedulerOpts) | ||||
| 	} else { | ||||
| 		scheduler = NewScheduler(opts.RedisConnOpt, opts.SchedulerOpts) | ||||
| 	} | ||||
|  | ||||
| 	syncInterval := opts.SyncInterval | ||||
| 	if syncInterval == 0 { | ||||
| 		syncInterval = defaultSyncInterval | ||||
| @@ -79,13 +89,13 @@ type PeriodicTaskConfig struct { | ||||
|  | ||||
| func (c *PeriodicTaskConfig) hash() string { | ||||
| 	h := sha256.New() | ||||
| 	io.WriteString(h, c.Cronspec) | ||||
| 	io.WriteString(h, c.Task.Type()) | ||||
| 	_, _ = h.Write([]byte(c.Cronspec)) | ||||
| 	_, _ = h.Write([]byte(c.Task.Type())) | ||||
| 	h.Write(c.Task.Payload()) | ||||
| 	opts := stringifyOptions(c.Opts) | ||||
| 	sort.Strings(opts) | ||||
| 	for _, opt := range opts { | ||||
| 		io.WriteString(h, opt) | ||||
| 		_, _ = h.Write([]byte(opt)) | ||||
| 	} | ||||
| 	return fmt.Sprintf("%x", h.Sum(nil)) | ||||
| } | ||||
| @@ -173,8 +183,8 @@ func (mgr *PeriodicTaskManager) add(configs []*PeriodicTaskConfig) { | ||||
| 	for _, c := range configs { | ||||
| 		entryID, err := mgr.s.Register(c.Cronspec, c.Task, c.Opts...) | ||||
| 		if err != nil { | ||||
| 			mgr.s.logger.Errorf("Failed to register periodic task: cronspec=%q task=%q", | ||||
| 				c.Cronspec, c.Task.Type()) | ||||
| 			mgr.s.logger.Errorf("Failed to register periodic task: cronspec=%q task=%q err=%v", | ||||
| 				c.Cronspec, c.Task.Type(), err) | ||||
| 			continue | ||||
| 		} | ||||
| 		mgr.m[c.hash()] = entryID | ||||
|   | ||||
| @@ -32,6 +32,7 @@ func (p *FakeConfigProvider) GetConfigs() ([]*PeriodicTaskConfig, error) { | ||||
| } | ||||
|  | ||||
| func TestNewPeriodicTaskManager(t *testing.T) { | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	cfgs := []*PeriodicTaskConfig{ | ||||
| 		{Cronspec: "* * * * *", Task: NewTask("foo", nil)}, | ||||
| 		{Cronspec: "* * * * *", Task: NewTask("bar", nil)}, | ||||
| @@ -43,14 +44,14 @@ func TestNewPeriodicTaskManager(t *testing.T) { | ||||
| 		{ | ||||
| 			desc: "with provider and redisConnOpt", | ||||
| 			opts: PeriodicTaskManagerOpts{ | ||||
| 				RedisConnOpt:               RedisClientOpt{Addr: ":6379"}, | ||||
| 				RedisConnOpt:               redisConnOpt, | ||||
| 				PeriodicTaskConfigProvider: &FakeConfigProvider{cfgs: cfgs}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "with sync option", | ||||
| 			opts: PeriodicTaskManagerOpts{ | ||||
| 				RedisConnOpt:               RedisClientOpt{Addr: ":6379"}, | ||||
| 				RedisConnOpt:               redisConnOpt, | ||||
| 				PeriodicTaskConfigProvider: &FakeConfigProvider{cfgs: cfgs}, | ||||
| 				SyncInterval:               5 * time.Minute, | ||||
| 			}, | ||||
| @@ -58,7 +59,7 @@ func TestNewPeriodicTaskManager(t *testing.T) { | ||||
| 		{ | ||||
| 			desc: "with scheduler option", | ||||
| 			opts: PeriodicTaskManagerOpts{ | ||||
| 				RedisConnOpt:               RedisClientOpt{Addr: ":6379"}, | ||||
| 				RedisConnOpt:               redisConnOpt, | ||||
| 				PeriodicTaskConfigProvider: &FakeConfigProvider{cfgs: cfgs}, | ||||
| 				SyncInterval:               5 * time.Minute, | ||||
| 				SchedulerOpts: &SchedulerOpts{ | ||||
| @@ -74,37 +75,33 @@ func TestNewPeriodicTaskManager(t *testing.T) { | ||||
| 			t.Errorf("%s; NewPeriodicTaskManager returned error: %v", tc.desc, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewPeriodicTaskManagerError(t *testing.T) { | ||||
| 	cfgs := []*PeriodicTaskConfig{ | ||||
| 		{Cronspec: "* * * * *", Task: NewTask("foo", nil)}, | ||||
| 		{Cronspec: "* * * * *", Task: NewTask("bar", nil)}, | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| 		opts PeriodicTaskManagerOpts | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "without provider", | ||||
| 			opts: PeriodicTaskManagerOpts{ | ||||
| 				RedisConnOpt: RedisClientOpt{Addr: ":6379"}, | ||||
| 	t.Run("error", func(t *testing.T) { | ||||
| 		tests := []struct { | ||||
| 			desc string | ||||
| 			opts PeriodicTaskManagerOpts | ||||
| 		}{ | ||||
| 			{ | ||||
| 				desc: "without provider", | ||||
| 				opts: PeriodicTaskManagerOpts{ | ||||
| 					RedisConnOpt: redisConnOpt, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "without redisConOpt", | ||||
| 			opts: PeriodicTaskManagerOpts{ | ||||
| 				PeriodicTaskConfigProvider: &FakeConfigProvider{cfgs: cfgs}, | ||||
| 			{ | ||||
| 				desc: "without redisConOpt", | ||||
| 				opts: PeriodicTaskManagerOpts{ | ||||
| 					PeriodicTaskConfigProvider: &FakeConfigProvider{cfgs: cfgs}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		_, err := NewPeriodicTaskManager(tc.opts) | ||||
| 		if err == nil { | ||||
| 			t.Errorf("%s; NewPeriodicTaskManager did not return error", tc.desc) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 		for _, tc := range tests { | ||||
| 			_, err := NewPeriodicTaskManager(tc.opts) | ||||
| 			if err == nil { | ||||
| 				t.Errorf("%s; NewPeriodicTaskManager did not return error", tc.desc) | ||||
| 			} | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestPeriodicTaskConfigHash(t *testing.T) { | ||||
|   | ||||
							
								
								
									
										133
									
								
								processor.go
									
									
									
									
									
								
							
							
						
						
									
										133
									
								
								processor.go
									
									
									
									
									
								
							| @@ -8,7 +8,7 @@ import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"math/rand" | ||||
| 	"math/rand/v2" | ||||
| 	"runtime" | ||||
| 	"runtime/debug" | ||||
| 	"sort" | ||||
| @@ -37,11 +37,11 @@ type processor struct { | ||||
| 	// orderedQueues is set only in strict-priority mode. | ||||
| 	orderedQueues []string | ||||
|  | ||||
| 	retryDelayFunc RetryDelayFunc | ||||
| 	isFailureFunc  func(error) bool | ||||
|  | ||||
| 	errHandler ErrorHandler | ||||
| 	taskCheckInterval time.Duration | ||||
| 	retryDelayFunc    RetryDelayFunc | ||||
| 	isFailureFunc     func(error) bool | ||||
|  | ||||
| 	errHandler      ErrorHandler | ||||
| 	shutdownTimeout time.Duration | ||||
|  | ||||
| 	// channel via which to send sync requests to syncer. | ||||
| @@ -73,20 +73,21 @@ type processor struct { | ||||
| } | ||||
|  | ||||
| type processorParams struct { | ||||
| 	logger          *log.Logger | ||||
| 	broker          base.Broker | ||||
| 	baseCtxFn       func() context.Context | ||||
| 	retryDelayFunc  RetryDelayFunc | ||||
| 	isFailureFunc   func(error) bool | ||||
| 	syncCh          chan<- *syncRequest | ||||
| 	cancelations    *base.Cancelations | ||||
| 	concurrency     int | ||||
| 	queues          map[string]int | ||||
| 	strictPriority  bool | ||||
| 	errHandler      ErrorHandler | ||||
| 	shutdownTimeout time.Duration | ||||
| 	starting        chan<- *workerInfo | ||||
| 	finished        chan<- *base.TaskMessage | ||||
| 	logger            *log.Logger | ||||
| 	broker            base.Broker | ||||
| 	baseCtxFn         func() context.Context | ||||
| 	retryDelayFunc    RetryDelayFunc | ||||
| 	taskCheckInterval time.Duration | ||||
| 	isFailureFunc     func(error) bool | ||||
| 	syncCh            chan<- *syncRequest | ||||
| 	cancelations      *base.Cancelations | ||||
| 	concurrency       int | ||||
| 	queues            map[string]int | ||||
| 	strictPriority    bool | ||||
| 	errHandler        ErrorHandler | ||||
| 	shutdownTimeout   time.Duration | ||||
| 	starting          chan<- *workerInfo | ||||
| 	finished          chan<- *base.TaskMessage | ||||
| } | ||||
|  | ||||
| // newProcessor constructs a new processor. | ||||
| @@ -97,26 +98,27 @@ func newProcessor(params processorParams) *processor { | ||||
| 		orderedQueues = sortByPriority(queues) | ||||
| 	} | ||||
| 	return &processor{ | ||||
| 		logger:          params.logger, | ||||
| 		broker:          params.broker, | ||||
| 		baseCtxFn:       params.baseCtxFn, | ||||
| 		clock:           timeutil.NewRealClock(), | ||||
| 		queueConfig:     queues, | ||||
| 		orderedQueues:   orderedQueues, | ||||
| 		retryDelayFunc:  params.retryDelayFunc, | ||||
| 		isFailureFunc:   params.isFailureFunc, | ||||
| 		syncRequestCh:   params.syncCh, | ||||
| 		cancelations:    params.cancelations, | ||||
| 		errLogLimiter:   rate.NewLimiter(rate.Every(3*time.Second), 1), | ||||
| 		sema:            make(chan struct{}, params.concurrency), | ||||
| 		done:            make(chan struct{}), | ||||
| 		quit:            make(chan struct{}), | ||||
| 		abort:           make(chan struct{}), | ||||
| 		errHandler:      params.errHandler, | ||||
| 		handler:         HandlerFunc(func(ctx context.Context, t *Task) error { return fmt.Errorf("handler not set") }), | ||||
| 		shutdownTimeout: params.shutdownTimeout, | ||||
| 		starting:        params.starting, | ||||
| 		finished:        params.finished, | ||||
| 		logger:            params.logger, | ||||
| 		broker:            params.broker, | ||||
| 		baseCtxFn:         params.baseCtxFn, | ||||
| 		clock:             timeutil.NewRealClock(), | ||||
| 		queueConfig:       queues, | ||||
| 		orderedQueues:     orderedQueues, | ||||
| 		taskCheckInterval: params.taskCheckInterval, | ||||
| 		retryDelayFunc:    params.retryDelayFunc, | ||||
| 		isFailureFunc:     params.isFailureFunc, | ||||
| 		syncRequestCh:     params.syncCh, | ||||
| 		cancelations:      params.cancelations, | ||||
| 		errLogLimiter:     rate.NewLimiter(rate.Every(3*time.Second), 1), | ||||
| 		sema:              make(chan struct{}, params.concurrency), | ||||
| 		done:              make(chan struct{}), | ||||
| 		quit:              make(chan struct{}), | ||||
| 		abort:             make(chan struct{}), | ||||
| 		errHandler:        params.errHandler, | ||||
| 		handler:           HandlerFunc(func(ctx context.Context, t *Task) error { return fmt.Errorf("handler not set") }), | ||||
| 		shutdownTimeout:   params.shutdownTimeout, | ||||
| 		starting:          params.starting, | ||||
| 		finished:          params.finished, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -179,7 +181,8 @@ func (p *processor) exec() { | ||||
| 			// Sleep to avoid slamming redis and let scheduler move tasks into queues. | ||||
| 			// Note: We are not using blocking pop operation and polling queues instead. | ||||
| 			// This adds significant load to redis. | ||||
| 			time.Sleep(time.Second) | ||||
| 			jitter := rand.N(p.taskCheckInterval) | ||||
| 			time.Sleep(p.taskCheckInterval/2 + jitter) | ||||
| 			<-p.sema // release token | ||||
| 			return | ||||
| 		case err != nil: | ||||
| @@ -259,7 +262,8 @@ func (p *processor) requeue(l *base.Lease, msg *base.TaskMessage) { | ||||
| 		// If lease is not valid, do not write to redis; Let recoverer take care of it. | ||||
| 		return | ||||
| 	} | ||||
| 	ctx, _ := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	ctx, cancel := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	defer cancel() | ||||
| 	err := p.broker.Requeue(ctx, msg) | ||||
| 	if err != nil { | ||||
| 		p.logger.Errorf("Could not push task id=%s back to queue: %v", msg.ID, err) | ||||
| @@ -281,7 +285,8 @@ func (p *processor) markAsComplete(l *base.Lease, msg *base.TaskMessage) { | ||||
| 		// If lease is not valid, do not write to redis; Let recoverer take care of it. | ||||
| 		return | ||||
| 	} | ||||
| 	ctx, _ := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	ctx, cancel := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	defer cancel() | ||||
| 	err := p.broker.MarkAsComplete(ctx, msg) | ||||
| 	if err != nil { | ||||
| 		errMsg := fmt.Sprintf("Could not move task id=%s type=%q from %q to %q:  %+v", | ||||
| @@ -302,7 +307,8 @@ func (p *processor) markAsDone(l *base.Lease, msg *base.TaskMessage) { | ||||
| 		// If lease is not valid, do not write to redis; Let recoverer take care of it. | ||||
| 		return | ||||
| 	} | ||||
| 	ctx, _ := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	ctx, cancel := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	defer cancel() | ||||
| 	err := p.broker.Done(ctx, msg) | ||||
| 	if err != nil { | ||||
| 		errMsg := fmt.Sprintf("Could not remove task id=%s type=%q from %q err: %+v", msg.ID, msg.Type, base.ActiveKey(msg.Queue), err) | ||||
| @@ -321,20 +327,23 @@ func (p *processor) markAsDone(l *base.Lease, msg *base.TaskMessage) { | ||||
| // the task should not be retried and should be archived instead. | ||||
| var SkipRetry = errors.New("skip retry for the task") | ||||
|  | ||||
| // RevokeTask is used as a return value from Handler.ProcessTask to indicate that | ||||
| // the task should not be retried or archived. | ||||
| var RevokeTask = errors.New("revoke task") | ||||
|  | ||||
| func (p *processor) handleFailedMessage(ctx context.Context, l *base.Lease, msg *base.TaskMessage, err error) { | ||||
| 	if p.errHandler != nil { | ||||
| 		p.errHandler.HandleError(ctx, NewTask(msg.Type, msg.Payload), err) | ||||
| 	} | ||||
| 	if !p.isFailureFunc(err) { | ||||
| 		// retry the task without marking it as failed | ||||
| 		p.retry(l, msg, err, false /*isFailure*/) | ||||
| 		return | ||||
| 	} | ||||
| 	if msg.Retried >= msg.Retry || errors.Is(err, SkipRetry) { | ||||
| 	switch { | ||||
| 	case errors.Is(err, RevokeTask): | ||||
| 		p.logger.Warnf("revoke task id=%s", msg.ID) | ||||
| 		p.markAsDone(l, msg) | ||||
| 	case msg.Retried >= msg.Retry || errors.Is(err, SkipRetry): | ||||
| 		p.logger.Warnf("Retry exhausted for task id=%s", msg.ID) | ||||
| 		p.archive(l, msg, err) | ||||
| 	} else { | ||||
| 		p.retry(l, msg, err, true /*isFailure*/) | ||||
| 	default: | ||||
| 		p.retry(l, msg, err, p.isFailureFunc(err)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -343,7 +352,8 @@ func (p *processor) retry(l *base.Lease, msg *base.TaskMessage, e error, isFailu | ||||
| 		// If lease is not valid, do not write to redis; Let recoverer take care of it. | ||||
| 		return | ||||
| 	} | ||||
| 	ctx, _ := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	ctx, cancel := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	defer cancel() | ||||
| 	d := p.retryDelayFunc(msg.Retried, e, NewTask(msg.Type, msg.Payload)) | ||||
| 	retryAt := time.Now().Add(d) | ||||
| 	err := p.broker.Retry(ctx, msg, retryAt, e.Error(), isFailure) | ||||
| @@ -365,7 +375,8 @@ func (p *processor) archive(l *base.Lease, msg *base.TaskMessage, e error) { | ||||
| 		// If lease is not valid, do not write to redis; Let recoverer take care of it. | ||||
| 		return | ||||
| 	} | ||||
| 	ctx, _ := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	ctx, cancel := context.WithDeadline(context.Background(), l.Deadline()) | ||||
| 	defer cancel() | ||||
| 	err := p.broker.Archive(ctx, msg, e.Error()) | ||||
| 	if err != nil { | ||||
| 		errMsg := fmt.Sprintf("Could not move task id=%s from %q to %q", msg.ID, base.ActiveKey(msg.Queue), base.ArchivedKey(msg.Queue)) | ||||
| @@ -402,8 +413,7 @@ func (p *processor) queues() []string { | ||||
| 			names = append(names, qname) | ||||
| 		} | ||||
| 	} | ||||
| 	r := rand.New(rand.NewSource(time.Now().UnixNano())) | ||||
| 	r.Shuffle(len(names), func(i, j int) { names[i], names[j] = names[j], names[i] }) | ||||
| 	rand.Shuffle(len(names), func(i, j int) { names[i], names[j] = names[j], names[i] }) | ||||
| 	return uniq(names, len(p.queueConfig)) | ||||
| } | ||||
|  | ||||
| @@ -420,12 +430,15 @@ func (p *processor) perform(ctx context.Context, task *Task) (err error) { | ||||
| 				// map/slice usage. The parent frame should have the real trigger. | ||||
| 				_, file, line, ok = runtime.Caller(2) | ||||
| 			} | ||||
|  | ||||
| 			var errMsg string | ||||
| 			// Include the file and line number info in the error, if runtime.Caller returned ok. | ||||
| 			if ok { | ||||
| 				err = fmt.Errorf("panic [%s:%d]: %v", file, line, x) | ||||
| 				errMsg = fmt.Sprintf("panic [%s:%d]: %v", file, line, x) | ||||
| 			} else { | ||||
| 				err = fmt.Errorf("panic: %v", x) | ||||
| 				errMsg = fmt.Sprintf("panic: %v", x) | ||||
| 			} | ||||
| 			err = &errors.PanicError{ | ||||
| 				ErrMsg: errMsg, | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| @@ -521,3 +534,7 @@ func (p *processor) computeDeadline(msg *base.TaskMessage) time.Time { | ||||
| 	} | ||||
| 	return time.Unix(msg.Deadline, 0) | ||||
| } | ||||
|  | ||||
| func IsPanicError(err error) bool { | ||||
| 	return errors.IsPanicError(err) | ||||
| } | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @@ -62,20 +63,21 @@ func newProcessorForTest(t *testing.T, r *rdb.RDB, h Handler) *processor { | ||||
| 	go fakeHeartbeater(starting, finished, done) | ||||
| 	go fakeSyncer(syncCh, done) | ||||
| 	p := newProcessor(processorParams{ | ||||
| 		logger:          testLogger, | ||||
| 		broker:          r, | ||||
| 		baseCtxFn:       context.Background, | ||||
| 		retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 		isFailureFunc:   defaultIsFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    base.NewCancelations(), | ||||
| 		concurrency:     10, | ||||
| 		queues:          defaultQueueConfig, | ||||
| 		strictPriority:  false, | ||||
| 		errHandler:      nil, | ||||
| 		shutdownTimeout: defaultShutdownTimeout, | ||||
| 		starting:        starting, | ||||
| 		finished:        finished, | ||||
| 		logger:            testLogger, | ||||
| 		broker:            r, | ||||
| 		baseCtxFn:         context.Background, | ||||
| 		retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 		taskCheckInterval: defaultTaskCheckInterval, | ||||
| 		isFailureFunc:     defaultIsFailureFunc, | ||||
| 		syncCh:            syncCh, | ||||
| 		cancelations:      base.NewCancelations(), | ||||
| 		concurrency:       10, | ||||
| 		queues:            defaultQueueConfig, | ||||
| 		strictPriority:    false, | ||||
| 		errHandler:        nil, | ||||
| 		shutdownTimeout:   defaultShutdownTimeout, | ||||
| 		starting:          starting, | ||||
| 		finished:          finished, | ||||
| 	}) | ||||
| 	p.handler = h | ||||
| 	return p | ||||
| @@ -293,6 +295,7 @@ func TestProcessorRetry(t *testing.T) { | ||||
|  | ||||
| 	errMsg := "something went wrong" | ||||
| 	wrappedSkipRetry := fmt.Errorf("%s:%w", errMsg, SkipRetry) | ||||
| 	wrappedRevokeTask := fmt.Errorf("%s:%w", errMsg, RevokeTask) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc         string              // test description | ||||
| @@ -310,7 +313,7 @@ func TestProcessorRetry(t *testing.T) { | ||||
| 			pending: []*base.TaskMessage{m1, m2, m3, m4}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return fmt.Errorf(errMsg) | ||||
| 				return errors.New(errMsg) | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantErrMsg:   errMsg, | ||||
| @@ -344,6 +347,32 @@ func TestProcessorRetry(t *testing.T) { | ||||
| 			wantArchived: []*base.TaskMessage{m1, m2}, | ||||
| 			wantErrCount: 2, // ErrorHandler should still be called with SkipRetry error | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should revoke task", | ||||
| 			pending: []*base.TaskMessage{m1, m2}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return RevokeTask // return RevokeTask without wrapping | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantErrMsg:   RevokeTask.Error(), | ||||
| 			wantRetry:    []*base.TaskMessage{}, | ||||
| 			wantArchived: []*base.TaskMessage{}, | ||||
| 			wantErrCount: 2, // ErrorHandler should still be called with RevokeTask error | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should revoke task (with error wrapping)", | ||||
| 			pending: []*base.TaskMessage{m1, m2}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return wrappedRevokeTask | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantErrMsg:   wrappedRevokeTask.Error(), | ||||
| 			wantRetry:    []*base.TaskMessage{}, | ||||
| 			wantArchived: []*base.TaskMessage{}, | ||||
| 			wantErrCount: 2, // ErrorHandler should still be called with RevokeTask error | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| @@ -538,20 +567,21 @@ func TestProcessorWithExpiredLease(t *testing.T) { | ||||
| 		}() | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			baseCtxFn:       context.Background, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:   defaultIsFailureFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          defaultQueueConfig, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 			logger:            testLogger, | ||||
| 			broker:            rdbClient, | ||||
| 			baseCtxFn:         context.Background, | ||||
| 			taskCheckInterval: defaultTaskCheckInterval, | ||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:     defaultIsFailureFunc, | ||||
| 			syncCh:            syncCh, | ||||
| 			cancelations:      base.NewCancelations(), | ||||
| 			concurrency:       10, | ||||
| 			queues:            defaultQueueConfig, | ||||
| 			strictPriority:    false, | ||||
| 			errHandler:        nil, | ||||
| 			shutdownTimeout:   defaultShutdownTimeout, | ||||
| 			starting:          starting, | ||||
| 			finished:          finished, | ||||
| 		}) | ||||
| 		p.handler = tc.handler | ||||
| 		var ( | ||||
| @@ -692,20 +722,21 @@ func TestProcessorWithStrictPriority(t *testing.T) { | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			baseCtxFn:       context.Background, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:   defaultIsFailureFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     1, // Set concurrency to 1 to make sure tasks are processed one at a time. | ||||
| 			queues:          queueCfg, | ||||
| 			strictPriority:  true, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 			logger:            testLogger, | ||||
| 			broker:            rdbClient, | ||||
| 			baseCtxFn:         context.Background, | ||||
| 			taskCheckInterval: defaultTaskCheckInterval, | ||||
| 			retryDelayFunc:    DefaultRetryDelayFunc, | ||||
| 			isFailureFunc:     defaultIsFailureFunc, | ||||
| 			syncCh:            syncCh, | ||||
| 			cancelations:      base.NewCancelations(), | ||||
| 			concurrency:       1, // Set concurrency to 1 to make sure tasks are processed one at a time. | ||||
| 			queues:            queueCfg, | ||||
| 			strictPriority:    true, | ||||
| 			errHandler:        nil, | ||||
| 			shutdownTimeout:   defaultShutdownTimeout, | ||||
| 			starting:          starting, | ||||
| 			finished:          finished, | ||||
| 		}) | ||||
| 		p.handler = HandlerFunc(handler) | ||||
|  | ||||
| @@ -921,3 +952,45 @@ func TestProcessorComputeDeadline(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestReturnPanicError(t *testing.T) { | ||||
|  | ||||
| 	task := NewTask("gen_thumbnail", h.JSON(map[string]interface{}{"src": "some/img/path"})) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name         string | ||||
| 		handler      HandlerFunc | ||||
| 		IsPanicError bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name: "should return panic error when occurred panic recovery", | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				panic("something went terribly wrong") | ||||
| 			}, | ||||
| 			IsPanicError: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name: "should return normal error when don't occur panic recovery", | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				return fmt.Errorf("something went terribly wrong") | ||||
| 			}, | ||||
| 			IsPanicError: false, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			p := processor{ | ||||
| 				logger:  log.NewLogger(nil), | ||||
| 				handler: tc.handler, | ||||
| 			} | ||||
| 			got := p.perform(context.Background(), task) | ||||
| 			if tc.IsPanicError != IsPanicError(got) { | ||||
| 				t.Errorf("%s: got=%t, want=%t", tc.name, IsPanicError(got), tc.IsPanicError) | ||||
| 			} | ||||
| 			if tc.IsPanicError && !strings.HasPrefix(got.Error(), "panic error cause by:") { | ||||
| 				t.Error("wrong text msg for panic error") | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										113
									
								
								scheduler.go
									
									
									
									
									
								
							
							
						
						
									
										113
									
								
								scheduler.go
									
									
									
									
									
								
							| @@ -10,11 +10,11 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/robfig/cron/v3" | ||||
| ) | ||||
|  | ||||
| @@ -26,16 +26,17 @@ type Scheduler struct { | ||||
|  | ||||
| 	state *serverState | ||||
|  | ||||
| 	logger          *log.Logger | ||||
| 	client          *Client | ||||
| 	rdb             *rdb.RDB | ||||
| 	cron            *cron.Cron | ||||
| 	location        *time.Location | ||||
| 	done            chan struct{} | ||||
| 	wg              sync.WaitGroup | ||||
| 	preEnqueueFunc  func(task *Task, opts []Option) | ||||
| 	postEnqueueFunc func(info *TaskInfo, err error) | ||||
| 	errHandler      func(task *Task, opts []Option, err error) | ||||
| 	heartbeatInterval time.Duration | ||||
| 	logger            *log.Logger | ||||
| 	client            *Client | ||||
| 	rdb               *rdb.RDB | ||||
| 	cron              *cron.Cron | ||||
| 	location          *time.Location | ||||
| 	done              chan struct{} | ||||
| 	wg                sync.WaitGroup | ||||
| 	preEnqueueFunc    func(task *Task, opts []Option) | ||||
| 	postEnqueueFunc   func(info *TaskInfo, err error) | ||||
| 	errHandler        func(task *Task, opts []Option, err error) | ||||
|  | ||||
| 	// guards idmap | ||||
| 	mu sync.Mutex | ||||
| @@ -45,17 +46,48 @@ type Scheduler struct { | ||||
| 	idmap map[string]cron.EntryID | ||||
| } | ||||
|  | ||||
| const defaultHeartbeatInterval = 10 * time.Second | ||||
|  | ||||
| // NewScheduler returns a new Scheduler instance given the redis connection option. | ||||
| // The parameter opts is optional, defaults will be used if opts is set to nil | ||||
| func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	scheduler := newScheduler(opts) | ||||
|  | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
|  | ||||
| 	rdb := rdb.NewRDB(redisClient) | ||||
|  | ||||
| 	scheduler.rdb = rdb | ||||
| 	scheduler.client = &Client{broker: rdb, 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 { | ||||
| 	scheduler := newScheduler(opts) | ||||
|  | ||||
| 	scheduler.rdb = rdb.NewRDB(c) | ||||
| 	scheduler.client = NewClientFromRedisClient(c) | ||||
|  | ||||
| 	return scheduler | ||||
| } | ||||
|  | ||||
| func newScheduler(opts *SchedulerOpts) *Scheduler { | ||||
| 	if opts == nil { | ||||
| 		opts = &SchedulerOpts{} | ||||
| 	} | ||||
|  | ||||
| 	heartbeatInterval := opts.HeartbeatInterval | ||||
| 	if heartbeatInterval <= 0 { | ||||
| 		heartbeatInterval = defaultHeartbeatInterval | ||||
| 	} | ||||
|  | ||||
| 	logger := log.NewLogger(opts.Logger) | ||||
| 	loglevel := opts.LogLevel | ||||
| 	if loglevel == level_unspecified { | ||||
| @@ -69,18 +101,17 @@ func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | ||||
| 	} | ||||
|  | ||||
| 	return &Scheduler{ | ||||
| 		id:              generateSchedulerID(), | ||||
| 		state:           &serverState{value: srvStateNew}, | ||||
| 		logger:          logger, | ||||
| 		client:          NewClient(r), | ||||
| 		rdb:             rdb.NewRDB(c), | ||||
| 		cron:            cron.New(cron.WithLocation(loc)), | ||||
| 		location:        loc, | ||||
| 		done:            make(chan struct{}), | ||||
| 		preEnqueueFunc:  opts.PreEnqueueFunc, | ||||
| 		postEnqueueFunc: opts.PostEnqueueFunc, | ||||
| 		errHandler:      opts.EnqueueErrorHandler, | ||||
| 		idmap:           make(map[string]cron.EntryID), | ||||
| 		id:                generateSchedulerID(), | ||||
| 		state:             &serverState{value: srvStateNew}, | ||||
| 		heartbeatInterval: heartbeatInterval, | ||||
| 		logger:            logger, | ||||
| 		cron:              cron.New(cron.WithLocation(loc)), | ||||
| 		location:          loc, | ||||
| 		done:              make(chan struct{}), | ||||
| 		preEnqueueFunc:    opts.PreEnqueueFunc, | ||||
| 		postEnqueueFunc:   opts.PostEnqueueFunc, | ||||
| 		errHandler:        opts.EnqueueErrorHandler, | ||||
| 		idmap:             make(map[string]cron.EntryID), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -94,6 +125,15 @@ func generateSchedulerID() string { | ||||
|  | ||||
| // SchedulerOpts specifies scheduler options. | ||||
| type SchedulerOpts struct { | ||||
| 	// HeartbeatInterval specifies the interval between scheduler heartbeats. | ||||
| 	// | ||||
| 	// If unset, zero or a negative value, the interval is set to 10 second. | ||||
| 	// | ||||
| 	// Note: Setting this value too low may add significant load to redis. | ||||
| 	// | ||||
| 	// By default, HeartbeatInterval is set to 10 seconds. | ||||
| 	HeartbeatInterval time.Duration | ||||
|  | ||||
| 	// Logger specifies the logger used by the scheduler instance. | ||||
| 	// | ||||
| 	// If unset, the default logger is used. | ||||
| @@ -261,19 +301,22 @@ func (s *Scheduler) Shutdown() { | ||||
| 	s.wg.Wait() | ||||
|  | ||||
| 	s.clearHistory() | ||||
| 	s.client.Close() | ||||
| 	s.rdb.Close() | ||||
| 	if err := s.client.Close(); err != nil { | ||||
| 		s.logger.Errorf("Failed to close redis client connection: %v", err) | ||||
| 	} | ||||
| 	s.logger.Info("Scheduler stopped") | ||||
| } | ||||
|  | ||||
| func (s *Scheduler) runHeartbeater() { | ||||
| 	defer s.wg.Done() | ||||
| 	ticker := time.NewTicker(5 * time.Second) | ||||
| 	ticker := time.NewTicker(s.heartbeatInterval) | ||||
| 	for { | ||||
| 		select { | ||||
| 		case <-s.done: | ||||
| 			s.logger.Debugf("Scheduler heatbeater shutting down") | ||||
| 			s.rdb.ClearSchedulerEntries(s.id) | ||||
| 			if err := s.rdb.ClearSchedulerEntries(s.id); err != nil { | ||||
| 				s.logger.Errorf("Failed to clear the scheduler entries: %v", err) | ||||
| 			} | ||||
| 			ticker.Stop() | ||||
| 			return | ||||
| 		case <-ticker.C: | ||||
| @@ -298,8 +341,7 @@ func (s *Scheduler) beat() { | ||||
| 		} | ||||
| 		entries = append(entries, e) | ||||
| 	} | ||||
| 	s.logger.Debugf("Writing entries %v", entries) | ||||
| 	if err := s.rdb.WriteSchedulerEntries(s.id, entries, 5*time.Second); err != nil { | ||||
| 	if err := s.rdb.WriteSchedulerEntries(s.id, entries, s.heartbeatInterval*2); err != nil { | ||||
| 		s.logger.Warnf("Scheduler could not write heartbeat data: %v", err) | ||||
| 	} | ||||
| } | ||||
| @@ -320,3 +362,14 @@ func (s *Scheduler) clearHistory() { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Ping performs a ping against the redis connection. | ||||
| func (s *Scheduler) Ping() error { | ||||
| 	s.state.mu.Lock() | ||||
| 	defer s.state.mu.Unlock() | ||||
| 	if s.state.value == srvStateClosed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	return s.rdb.Ping() | ||||
| } | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/testutil" | ||||
| @@ -58,6 +59,7 @@ func TestSchedulerRegister(t *testing.T) { | ||||
|  | ||||
| 	r := setup(t) | ||||
|  | ||||
| 	// Tests for new redis connection. | ||||
| 	for _, tc := range tests { | ||||
| 		scheduler := NewScheduler(getRedisConnOpt(t), nil) | ||||
| 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | ||||
| @@ -75,6 +77,28 @@ func TestSchedulerRegister(t *testing.T) { | ||||
| 			t.Errorf("mismatch found in queue %q: (-want,+got)\n%s", tc.queue, diff) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	r = setup(t) | ||||
|  | ||||
| 	// Tests for existing redis connection. | ||||
| 	for _, tc := range tests { | ||||
| 		redisClient := getRedisConnOpt(t).MakeRedisClient().(redis.UniversalClient) | ||||
| 		scheduler := NewSchedulerFromRedisClient(redisClient, nil) | ||||
| 		if _, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		if err := scheduler.Start(); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		time.Sleep(tc.wait) | ||||
| 		scheduler.Shutdown() | ||||
|  | ||||
| 		got := testutil.GetPendingMessages(t, r, tc.queue) | ||||
| 		if diff := cmp.Diff(tc.want, got, testutil.IgnoreIDOpt); diff != "" { | ||||
| 			t.Errorf("mismatch found in queue %q: (-want,+got)\n%s", tc.queue, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestSchedulerWhenRedisDown(t *testing.T) { | ||||
| @@ -90,7 +114,7 @@ func TestSchedulerWhenRedisDown(t *testing.T) { | ||||
|  | ||||
| 	// Connect to non-existent redis instance to simulate a redis server being down. | ||||
| 	scheduler := NewScheduler( | ||||
| 		RedisClientOpt{Addr: ":9876"}, | ||||
| 		RedisClientOpt{Addr: ":9876"}, // no Redis listening to this port. | ||||
| 		&SchedulerOpts{EnqueueErrorHandler: errorHandler}, | ||||
| 	) | ||||
|  | ||||
|   | ||||
| @@ -144,9 +144,7 @@ func (mux *ServeMux) HandleFunc(pattern string, handler func(context.Context, *T | ||||
| func (mux *ServeMux) Use(mws ...MiddlewareFunc) { | ||||
| 	mux.mu.Lock() | ||||
| 	defer mux.mu.Unlock() | ||||
| 	for _, fn := range mws { | ||||
| 		mux.mws = append(mux.mws, fn) | ||||
| 	} | ||||
| 	mux.mws = append(mux.mws, mws...) | ||||
| } | ||||
|  | ||||
| // NotFound returns an error indicating that the handler was not found for the given task. | ||||
|   | ||||
							
								
								
									
										163
									
								
								server.go
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								server.go
									
									
									
									
									
								
							| @@ -9,16 +9,16 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"math/rand" | ||||
| 	"math/rand/v2" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // Server is responsible for task processing and task lifecycle management. | ||||
| @@ -37,6 +37,9 @@ type Server struct { | ||||
| 	logger *log.Logger | ||||
|  | ||||
| 	broker base.Broker | ||||
| 	// When a Server has been created with an existing Redis connection, we do | ||||
| 	// not want to close it. | ||||
| 	sharedConnection bool | ||||
|  | ||||
| 	state *serverState | ||||
|  | ||||
| @@ -104,6 +107,15 @@ type Config struct { | ||||
| 	// If this is defined, then it MUST return a non-nil context | ||||
| 	BaseContext func() context.Context | ||||
|  | ||||
| 	// TaskCheckInterval specifies the interval between checks for new tasks to process when all queues are empty. | ||||
| 	// | ||||
| 	// If unset, zero or a negative value, the interval is set to 1 second. | ||||
| 	// | ||||
| 	// Note: Setting this value too low may add significant load to redis. | ||||
| 	// | ||||
| 	// By default, TaskCheckInterval is set to 1 seconds. | ||||
| 	TaskCheckInterval time.Duration | ||||
|  | ||||
| 	// Function to calculate retry delay for a failed task. | ||||
| 	// | ||||
| 	// By default, it uses exponential backoff algorithm to calculate the delay. | ||||
| @@ -162,6 +174,16 @@ type Config struct { | ||||
| 	//     }) | ||||
| 	// | ||||
| 	//     ErrorHandler: asynq.ErrorHandlerFunc(reportError) | ||||
|  | ||||
| 	//    we can also handle panic error like: | ||||
| 	//     func reportError(ctx context, task *asynq.Task, err error) { | ||||
| 	//         if asynq.IsPanic(err) { | ||||
| 	//	          errorReportingService.Notify(err) | ||||
| 	// 	       } | ||||
| 	//     }) | ||||
| 	// | ||||
| 	//     ErrorHandler: asynq.ErrorHandlerFunc(reportError) | ||||
|  | ||||
| 	ErrorHandler ErrorHandler | ||||
|  | ||||
| 	// Logger specifies the logger used by the server instance. | ||||
| @@ -220,6 +242,17 @@ type Config struct { | ||||
| 	// | ||||
| 	// If unset or nil, the group aggregation feature will be disabled on the server. | ||||
| 	GroupAggregator GroupAggregator | ||||
|  | ||||
| 	// JanitorInterval specifies the average interval of janitor checks for expired completed tasks. | ||||
| 	// | ||||
| 	// If unset or zero, default interval of 8 seconds is used. | ||||
| 	JanitorInterval time.Duration | ||||
|  | ||||
| 	// JanitorBatchSize specifies the number of expired completed tasks to be deleted in one run. | ||||
| 	// | ||||
| 	// If unset or zero, default batch size of 100 is used. | ||||
| 	// Make sure to not put a big number as the batch size to prevent a long-running script. | ||||
| 	JanitorBatchSize int | ||||
| } | ||||
|  | ||||
| // GroupAggregator aggregates a group of tasks into one before the tasks are passed to the Handler. | ||||
| @@ -367,9 +400,8 @@ func toInternalLogLevel(l LogLevel) log.Level { | ||||
| // DefaultRetryDelayFunc is the default RetryDelayFunc used if one is not specified in Config. | ||||
| // It uses exponential back-off strategy to calculate the retry delay. | ||||
| func DefaultRetryDelayFunc(n int, e error, t *Task) time.Duration { | ||||
| 	r := rand.New(rand.NewSource(time.Now().UnixNano())) | ||||
| 	// Formula taken from https://github.com/mperham/sidekiq. | ||||
| 	s := int(math.Pow(float64(n), 4)) + 15 + (r.Intn(30) * (n + 1)) | ||||
| 	s := int(math.Pow(float64(n), 4)) + 15 + (rand.IntN(30) * (n + 1)) | ||||
| 	return time.Duration(s) * time.Second | ||||
| } | ||||
|  | ||||
| @@ -380,6 +412,8 @@ var defaultQueueConfig = map[string]int{ | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	defaultTaskCheckInterval = 1 * time.Second | ||||
|  | ||||
| 	defaultShutdownTimeout = 8 * time.Second | ||||
|  | ||||
| 	defaultHealthCheckInterval = 15 * time.Second | ||||
| @@ -387,15 +421,28 @@ const ( | ||||
| 	defaultDelayedTaskCheckInterval = 5 * time.Second | ||||
|  | ||||
| 	defaultGroupGracePeriod = 1 * time.Minute | ||||
|  | ||||
| 	defaultJanitorInterval = 8 * time.Second | ||||
|  | ||||
| 	defaultJanitorBatchSize = 100 | ||||
| ) | ||||
|  | ||||
| // NewServer returns a new Server given a redis connection option | ||||
| // and server configuration. | ||||
| func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	redisClient, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	server := NewServerFromRedisClient(redisClient, cfg) | ||||
| 	server.sharedConnection = false | ||||
| 	return server | ||||
| } | ||||
|  | ||||
| // NewServerFromRedisClient returns a new instance of Server given a redis.UniversalClient | ||||
| // and server configuration | ||||
| // Warning: The underlying redis connection pool will not be closed by Asynq, you are responsible for closing it. | ||||
| func NewServerFromRedisClient(c redis.UniversalClient, cfg Config) *Server { | ||||
| 	baseCtxFn := cfg.BaseContext | ||||
| 	if baseCtxFn == nil { | ||||
| 		baseCtxFn = context.Background | ||||
| @@ -404,6 +451,12 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	if n < 1 { | ||||
| 		n = runtime.NumCPU() | ||||
| 	} | ||||
|  | ||||
| 	taskCheckInterval := cfg.TaskCheckInterval | ||||
| 	if taskCheckInterval <= 0 { | ||||
| 		taskCheckInterval = defaultTaskCheckInterval | ||||
| 	} | ||||
|  | ||||
| 	delayFunc := cfg.RetryDelayFunc | ||||
| 	if delayFunc == nil { | ||||
| 		delayFunc = DefaultRetryDelayFunc | ||||
| @@ -490,20 +543,21 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		cancelations: cancels, | ||||
| 	}) | ||||
| 	processor := newProcessor(processorParams{ | ||||
| 		logger:          logger, | ||||
| 		broker:          rdb, | ||||
| 		retryDelayFunc:  delayFunc, | ||||
| 		baseCtxFn:       baseCtxFn, | ||||
| 		isFailureFunc:   isFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    cancels, | ||||
| 		concurrency:     n, | ||||
| 		queues:          queues, | ||||
| 		strictPriority:  cfg.StrictPriority, | ||||
| 		errHandler:      cfg.ErrorHandler, | ||||
| 		shutdownTimeout: shutdownTimeout, | ||||
| 		starting:        starting, | ||||
| 		finished:        finished, | ||||
| 		logger:            logger, | ||||
| 		broker:            rdb, | ||||
| 		retryDelayFunc:    delayFunc, | ||||
| 		taskCheckInterval: taskCheckInterval, | ||||
| 		baseCtxFn:         baseCtxFn, | ||||
| 		isFailureFunc:     isFailureFunc, | ||||
| 		syncCh:            syncCh, | ||||
| 		cancelations:      cancels, | ||||
| 		concurrency:       n, | ||||
| 		queues:            queues, | ||||
| 		strictPriority:    cfg.StrictPriority, | ||||
| 		errHandler:        cfg.ErrorHandler, | ||||
| 		shutdownTimeout:   shutdownTimeout, | ||||
| 		starting:          starting, | ||||
| 		finished:          finished, | ||||
| 	}) | ||||
| 	recoverer := newRecoverer(recovererParams{ | ||||
| 		logger:         logger, | ||||
| @@ -519,11 +573,26 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		interval:        healthcheckInterval, | ||||
| 		healthcheckFunc: cfg.HealthCheckFunc, | ||||
| 	}) | ||||
|  | ||||
| 	janitorInterval := cfg.JanitorInterval | ||||
| 	if janitorInterval == 0 { | ||||
| 		janitorInterval = defaultJanitorInterval | ||||
| 	} | ||||
|  | ||||
| 	janitorBatchSize := cfg.JanitorBatchSize | ||||
| 	if janitorBatchSize == 0 { | ||||
| 		janitorBatchSize = defaultJanitorBatchSize | ||||
| 	} | ||||
| 	if janitorBatchSize > defaultJanitorBatchSize { | ||||
| 		logger.Warnf("Janitor batch size of %d is greater than the recommended batch size of %d. "+ | ||||
| 			"This might cause a long-running script", janitorBatchSize, defaultJanitorBatchSize) | ||||
| 	} | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:   logger, | ||||
| 		broker:   rdb, | ||||
| 		queues:   qnames, | ||||
| 		interval: 8 * time.Second, | ||||
| 		logger:    logger, | ||||
| 		broker:    rdb, | ||||
| 		queues:    qnames, | ||||
| 		interval:  janitorInterval, | ||||
| 		batchSize: janitorBatchSize, | ||||
| 	}) | ||||
| 	aggregator := newAggregator(aggregatorParams{ | ||||
| 		logger:          logger, | ||||
| @@ -535,18 +604,19 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		groupAggregator: cfg.GroupAggregator, | ||||
| 	}) | ||||
| 	return &Server{ | ||||
| 		logger:        logger, | ||||
| 		broker:        rdb, | ||||
| 		state:         srvState, | ||||
| 		forwarder:     forwarder, | ||||
| 		processor:     processor, | ||||
| 		syncer:        syncer, | ||||
| 		heartbeater:   heartbeater, | ||||
| 		subscriber:    subscriber, | ||||
| 		recoverer:     recoverer, | ||||
| 		healthchecker: healthchecker, | ||||
| 		janitor:       janitor, | ||||
| 		aggregator:    aggregator, | ||||
| 		logger:           logger, | ||||
| 		broker:           rdb, | ||||
| 		sharedConnection: true, | ||||
| 		state:            srvState, | ||||
| 		forwarder:        forwarder, | ||||
| 		processor:        processor, | ||||
| 		syncer:           syncer, | ||||
| 		heartbeater:      heartbeater, | ||||
| 		subscriber:       subscriber, | ||||
| 		recoverer:        recoverer, | ||||
| 		healthchecker:    healthchecker, | ||||
| 		janitor:          janitor, | ||||
| 		aggregator:       aggregator, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -562,6 +632,10 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| // One exception to this rule is when ProcessTask returns a SkipRetry error. | ||||
| // If the returned error is SkipRetry or an error wraps SkipRetry, retry is | ||||
| // skipped and the task will be immediately archived instead. | ||||
| // | ||||
| // Another exception to this rule is when ProcessTask returns a RevokeTask error. | ||||
| // If the returned error is RevokeTask or an error wraps RevokeTask, the task | ||||
| // will not be retried or archived. | ||||
| type Handler interface { | ||||
| 	ProcessTask(context.Context, *Task) error | ||||
| } | ||||
| @@ -674,7 +748,9 @@ func (srv *Server) Shutdown() { | ||||
| 	srv.heartbeater.shutdown() | ||||
| 	srv.wg.Wait() | ||||
|  | ||||
| 	srv.broker.Close() | ||||
| 	if !srv.sharedConnection { | ||||
| 		srv.broker.Close() | ||||
| 	} | ||||
| 	srv.logger.Info("Exiting") | ||||
| } | ||||
|  | ||||
| @@ -686,7 +762,7 @@ func (srv *Server) Shutdown() { | ||||
| func (srv *Server) Stop() { | ||||
| 	srv.state.mu.Lock() | ||||
| 	if srv.state.value != srvStateActive { | ||||
| 		// Invalid calll to Stop, server can only go from Active state to Stopped state. | ||||
| 		// Invalid call to Stop, server can only go from Active state to Stopped state. | ||||
| 		srv.state.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| @@ -697,3 +773,16 @@ func (srv *Server) Stop() { | ||||
| 	srv.processor.stop() | ||||
| 	srv.logger.Info("Processor stopped") | ||||
| } | ||||
|  | ||||
| // Ping performs a ping against the redis connection. | ||||
| // | ||||
| // This is an alternative to the HealthCheckFunc available in the Config object. | ||||
| func (srv *Server) Ping() error { | ||||
| 	srv.state.mu.Lock() | ||||
| 	defer srv.state.mu.Unlock() | ||||
| 	if srv.state.value == srvStateClosed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	return srv.broker.Ping() | ||||
| } | ||||
|   | ||||
| @@ -14,22 +14,12 @@ import ( | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/hibiken/asynq/internal/testbroker" | ||||
| 	"github.com/hibiken/asynq/internal/testutil" | ||||
|  | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"go.uber.org/goleak" | ||||
| ) | ||||
|  | ||||
| func TestServer(t *testing.T) { | ||||
| 	// 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, | ||||
| 	}) | ||||
|  | ||||
| func testServer(t *testing.T, c *Client, srv *Server) { | ||||
| 	// no-op handler | ||||
| 	h := func(ctx context.Context, task *Task) error { | ||||
| 		return nil | ||||
| @@ -53,18 +43,55 @@ func TestServer(t *testing.T) { | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestServer(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	c := NewClient(redisConnOpt) | ||||
| 	defer c.Close() | ||||
| 	srv := NewServer(redisConnOpt, Config{ | ||||
| 		Concurrency: 10, | ||||
| 		LogLevel:    testLogLevel, | ||||
| 	}) | ||||
|  | ||||
| 	testServer(t, c, srv) | ||||
| } | ||||
|  | ||||
| func TestServerFromRedisClient(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| 	redisClient := redisConnOpt.MakeRedisClient().(redis.UniversalClient) | ||||
| 	c := NewClientFromRedisClient(redisClient) | ||||
| 	srv := NewServerFromRedisClient(redisClient, Config{ | ||||
| 		Concurrency: 10, | ||||
| 		LogLevel:    testLogLevel, | ||||
| 	}) | ||||
|  | ||||
| 	testServer(t, c, srv) | ||||
|  | ||||
| 	err := c.Close() | ||||
| 	if err == nil { | ||||
| 		t.Error("client.Close() should have failed because of a shared client but it didn't") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestServerRun(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/redis/go-redis/v9/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNone(t, ignoreOpt) | ||||
|  | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	srv := NewServer(getRedisConnOpt(t), Config{LogLevel: testLogLevel}) | ||||
|  | ||||
| 	done := make(chan struct{}) | ||||
| 	// Make sure server exits when receiving TERM signal. | ||||
| 	go func() { | ||||
| 		time.Sleep(2 * time.Second) | ||||
| 		syscall.Kill(syscall.Getpid(), syscall.SIGTERM) | ||||
| 		_ = syscall.Kill(syscall.Getpid(), syscall.SIGTERM) | ||||
| 		done <- struct{}{} | ||||
| 	}() | ||||
|  | ||||
| @@ -83,7 +110,7 @@ func TestServerRun(t *testing.T) { | ||||
| } | ||||
|  | ||||
| func TestServerErrServerClosed(t *testing.T) { | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	srv := NewServer(getRedisConnOpt(t), Config{LogLevel: testLogLevel}) | ||||
| 	handler := NewServeMux() | ||||
| 	if err := srv.Start(handler); err != nil { | ||||
| 		t.Fatal(err) | ||||
| @@ -96,7 +123,7 @@ func TestServerErrServerClosed(t *testing.T) { | ||||
| } | ||||
|  | ||||
| func TestServerErrNilHandler(t *testing.T) { | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	srv := NewServer(getRedisConnOpt(t), Config{LogLevel: testLogLevel}) | ||||
| 	err := srv.Start(nil) | ||||
| 	if err == nil { | ||||
| 		t.Error("Starting server with nil handler: (*Server).Start(nil) did not return error") | ||||
| @@ -105,7 +132,7 @@ func TestServerErrNilHandler(t *testing.T) { | ||||
| } | ||||
|  | ||||
| func TestServerErrServerRunning(t *testing.T) { | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	srv := NewServer(getRedisConnOpt(t), Config{LogLevel: testLogLevel}) | ||||
| 	handler := NewServeMux() | ||||
| 	if err := srv.Start(handler); err != nil { | ||||
| 		t.Fatal(err) | ||||
| @@ -126,7 +153,7 @@ func TestServerWithRedisDown(t *testing.T) { | ||||
| 	}() | ||||
| 	r := rdb.NewRDB(setup(t)) | ||||
| 	testBroker := testbroker.NewTestBroker(r) | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	srv := NewServer(getRedisConnOpt(t), Config{LogLevel: testLogLevel}) | ||||
| 	srv.broker = testBroker | ||||
| 	srv.forwarder.broker = testBroker | ||||
| 	srv.heartbeater.broker = testBroker | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| // +build linux bsd darwin | ||||
| //go:build linux || dragonfly || freebsd || netbsd || openbsd || darwin | ||||
|  | ||||
| package asynq | ||||
|  | ||||
| @@ -24,8 +24,10 @@ func (srv *Server) waitForSignals() { | ||||
| 		if sig == unix.SIGTSTP { | ||||
| 			srv.Stop() | ||||
| 			continue | ||||
| 		} else { | ||||
| 			srv.Stop() | ||||
| 			break | ||||
| 		} | ||||
| 		break | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| // +build windows | ||||
| //go:build windows | ||||
|  | ||||
| package asynq | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ Asynq CLI is a command line tool to monitor the queues and tasks managed by `asy | ||||
|  | ||||
| In order to use the tool, compile it using the following command: | ||||
|  | ||||
|     go install github.com/hibiken/asynq/tools/asynq | ||||
|     go install github.com/hibiken/asynq/tools/asynq@latest | ||||
|  | ||||
| This will create the asynq executable under your `$GOPATH/bin` directory. | ||||
|  | ||||
| @@ -25,7 +25,7 @@ To view details on any command, use `asynq help <command> <subcommand>`. | ||||
| - `asynq dash` | ||||
| - `asynq stats` | ||||
| - `asynq queue [ls inspect history rm pause unpause]` | ||||
| - `asynq task [ls cancel delete archive run delete-all archive-all run-all]` | ||||
| - `asynq task [ls cancel delete archive run deleteall archiveall runall]` | ||||
| - `asynq server [ls]` | ||||
|  | ||||
| ### Global flags | ||||
|   | ||||
| @@ -11,15 +11,16 @@ import ( | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"text/tabwriter" | ||||
| 	"time" | ||||
| 	"unicode" | ||||
| 	"unicode/utf8" | ||||
|  | ||||
| 	"github.com/MakeNowJust/heredoc/v2" | ||||
| 	"github.com/fatih/color" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| 	"github.com/spf13/cobra" | ||||
| 	"github.com/spf13/pflag" | ||||
| 	"golang.org/x/exp/utf8string" | ||||
| @@ -39,6 +40,7 @@ var ( | ||||
| 	useRedisCluster bool | ||||
| 	clusterAddrs    string | ||||
| 	tlsServerName   string | ||||
| 	insecure        bool | ||||
| ) | ||||
|  | ||||
| // rootCmd represents the base command when called without any subcommands | ||||
| @@ -313,6 +315,8 @@ func init() { | ||||
| 		"List of comma-separated redis server addresses") | ||||
| 	rootCmd.PersistentFlags().StringVar(&tlsServerName, "tls_server", | ||||
| 		"", "Server name for TLS validation") | ||||
| 	rootCmd.PersistentFlags().BoolVar(&insecure, "insecure", | ||||
| 		false, "Allow insecure TLS connection by skipping cert validation") | ||||
| 	// Bind flags with config. | ||||
| 	viper.BindPFlag("uri", rootCmd.PersistentFlags().Lookup("uri")) | ||||
| 	viper.BindPFlag("db", rootCmd.PersistentFlags().Lookup("db")) | ||||
| @@ -320,6 +324,7 @@ func init() { | ||||
| 	viper.BindPFlag("cluster", rootCmd.PersistentFlags().Lookup("cluster")) | ||||
| 	viper.BindPFlag("cluster_addrs", rootCmd.PersistentFlags().Lookup("cluster_addrs")) | ||||
| 	viper.BindPFlag("tls_server", rootCmd.PersistentFlags().Lookup("tls_server")) | ||||
| 	viper.BindPFlag("insecure", rootCmd.PersistentFlags().Lookup("insecure")) | ||||
| } | ||||
|  | ||||
| // initConfig reads in config file and ENV variables if set. | ||||
| @@ -369,7 +374,12 @@ func createRDB() *rdb.RDB { | ||||
| 	return rdb.NewRDB(c) | ||||
| } | ||||
|  | ||||
| // createRDB creates a Inspector instance using flag values and returns it. | ||||
| // createClient creates a Client instance using flag values and returns it. | ||||
| func createClient() *asynq.Client { | ||||
| 	return asynq.NewClient(getRedisConnOpt()) | ||||
| } | ||||
|  | ||||
| // createInspector creates a Inspector instance using flag values and returns it. | ||||
| func createInspector() *asynq.Inspector { | ||||
| 	return asynq.NewInspector(getRedisConnOpt()) | ||||
| } | ||||
| @@ -396,7 +406,7 @@ func getTLSConfig() *tls.Config { | ||||
| 	if tlsServer == "" { | ||||
| 		return nil | ||||
| 	} | ||||
| 	return &tls.Config{ServerName: tlsServer} | ||||
| 	return &tls.Config{ServerName: tlsServer, InsecureSkipVerify: viper.GetBool("insecure")} | ||||
| } | ||||
|  | ||||
| // printTable is a helper function to print data in table format. | ||||
| @@ -456,3 +466,37 @@ func isPrintable(data []byte) bool { | ||||
| 	} | ||||
| 	return !isAllSpace | ||||
| } | ||||
|  | ||||
| // Helper to turn a command line flag into a duration | ||||
| func getDuration(cmd *cobra.Command, arg string) time.Duration { | ||||
| 	durationStr, err := cmd.Flags().GetString(arg) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	duration, err := time.ParseDuration(durationStr) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	return duration | ||||
| } | ||||
|  | ||||
| // Helper to turn a command line flag into a time | ||||
| func getTime(cmd *cobra.Command, arg string) time.Time { | ||||
| 	timeStr, err := cmd.Flags().GetString(arg) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	timeVal, err := time.Parse(time.RFC3339, timeStr) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	return timeVal | ||||
| } | ||||
|   | ||||
| @@ -53,6 +53,24 @@ func init() { | ||||
| 	taskRunCmd.MarkFlagRequired("queue") | ||||
| 	taskRunCmd.MarkFlagRequired("id") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskEnqueueCmd) | ||||
| 	taskEnqueueCmd.Flags().StringP("type_name", "t", "", "type name to enqueue the task as (required)") | ||||
| 	taskEnqueueCmd.Flags().StringP("payload", "l", "", "payload to enqueue (required)") | ||||
| 	// The following are the various OptionTypes; if not specified we won't pass them so that composeOptions() | ||||
| 	// can apply its own defaults | ||||
| 	taskEnqueueCmd.Flags().Int("retry", 0, "maximum retries") | ||||
| 	taskEnqueueCmd.Flags().String("queue", "", "queue to enqueue the task to") | ||||
| 	taskEnqueueCmd.Flags().String("id", "", "id to enqueue the task as") | ||||
| 	taskEnqueueCmd.Flags().String("timeout", "", "timeout for the task (how long it can run); must be parseable as a time.Duration") | ||||
| 	taskEnqueueCmd.Flags().String("deadline", "", "deadline for the task; must be in RFC3339 format") | ||||
| 	taskEnqueueCmd.Flags().String("unique", "", "unique period for the task (duration within which it is guaranteed to be unique); must be parseable as a time.Duration") | ||||
| 	taskEnqueueCmd.Flags().String("process_at", "", "process at time for the task; must be in RFC3339 format") | ||||
| 	taskEnqueueCmd.Flags().String("process_in", "", "process in window for the task; must be parseable as a time.Duration") | ||||
| 	taskEnqueueCmd.Flags().String("retention", "", "retention window for the task; must be parseable as a time.Duration") | ||||
| 	taskEnqueueCmd.Flags().String("group", "", "group for the task") | ||||
| 	taskEnqueueCmd.MarkFlagRequired("type_name") | ||||
| 	taskEnqueueCmd.MarkFlagRequired("payload") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskArchiveAllCmd) | ||||
| 	taskArchiveAllCmd.Flags().StringP("queue", "q", "", "queue to which the tasks belong (required)") | ||||
| 	taskArchiveAllCmd.Flags().StringP("state", "s", "", "state of the tasks; one of { pending | aggregating | scheduled | retry } (required)") | ||||
| @@ -151,6 +169,16 @@ var taskRunCmd = &cobra.Command{ | ||||
| 		$ asynq task run --queue=myqueue --id=f1720682-f5a6-4db1-8953-4f48ae541d0f`), | ||||
| } | ||||
|  | ||||
| var taskEnqueueCmd = &cobra.Command{ | ||||
| 	Use:   "enqueue --type_name=footype --payload=barpayload", | ||||
| 	Short: "Enqueue a task", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskEnqueue, | ||||
| 	Example: heredoc.Doc(` | ||||
| 		$ asynq task enqueue -t footype -l barpayload | ||||
| 		$ asynq task enqueue -t footask -l barpayload --retry 3 --id f1720682-f5a6-4db1-8953-4f48ae541d0f --queue bazqueue --timeout 100s --deadline 2024-12-14T01:23:45Z --unique 100s --process_at 2024-12-14T01:22:05Z --process_in 100s --retention 5h --group baygroup`), | ||||
| } | ||||
|  | ||||
| var taskArchiveAllCmd = &cobra.Command{ | ||||
| 	Use:   "archiveall --queue=<queue> --state=<state>", | ||||
| 	Short: "Archive all tasks in the given state", | ||||
| @@ -521,6 +549,95 @@ func taskRun(cmd *cobra.Command, args []string) { | ||||
| 	fmt.Println("task is now pending") | ||||
| } | ||||
|  | ||||
| func taskEnqueue(cmd *cobra.Command, args []string) { | ||||
| 	typeName, err := cmd.Flags().GetString("type_name") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	payload, err := cmd.Flags().GetString("payload") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	// For all of the optional flags, we need to explicitly check whether they were set or | ||||
| 	// not; for consistency we want to use the defaults set in composeOptions() rather than | ||||
| 	// the ones in the flag definitions. | ||||
| 	opts := []asynq.Option{} | ||||
| 	if cmd.Flags().Changed("retry") { | ||||
| 		retry, err := cmd.Flags().GetInt("retry") | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 		opts = append(opts, asynq.MaxRetry(retry)) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("queue") { | ||||
| 		queue, err := cmd.Flags().GetString("queue") | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 		opts = append(opts, asynq.Queue(queue)) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("id") { | ||||
| 		id, err := cmd.Flags().GetString("id") | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 		opts = append(opts, asynq.TaskID(id)) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("timeout") { | ||||
| 		opts = append(opts, asynq.Timeout(getDuration(cmd, "timeout"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("deadline") { | ||||
| 		opts = append(opts, asynq.Deadline(getTime(cmd, "deadline"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("unique") { | ||||
| 		opts = append(opts, asynq.Unique(getDuration(cmd, "unique"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("process_at") { | ||||
| 		opts = append(opts, asynq.ProcessAt(getTime(cmd, "process_at"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("process_in") { | ||||
| 		opts = append(opts, asynq.ProcessIn(getDuration(cmd, "process_in"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("retention") { | ||||
| 		opts = append(opts, asynq.Retention(getDuration(cmd, "retention"))) | ||||
| 	} | ||||
|  | ||||
| 	if cmd.Flags().Changed("group") { | ||||
| 		group, err := cmd.Flags().GetString("group") | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 		opts = append(opts, asynq.Group(group)) | ||||
| 	} | ||||
|  | ||||
| 	c := createClient() | ||||
| 	task := asynq.NewTask(typeName, []byte(payload), opts...) | ||||
|  | ||||
| 	taskInfo, err := c.Enqueue(task) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	fmt.Printf("Enqueued task %s to queue %s\n", taskInfo.ID, taskInfo.Queue) | ||||
| } | ||||
|  | ||||
| func taskArchiveAll(cmd *cobra.Command, args []string) { | ||||
| 	qname, err := cmd.Flags().GetString("queue") | ||||
| 	if err != nil { | ||||
| @@ -653,3 +770,4 @@ func taskRunAll(cmd *cobra.Command, args []string) { | ||||
| 	} | ||||
| 	fmt.Printf("%d tasks are now pending\n", n) | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										32
									
								
								tools/go.mod
									
									
									
									
									
								
							
							
						
						
									
										32
									
								
								tools/go.mod
									
									
									
									
									
								
							| @@ -1,18 +1,18 @@ | ||||
| module github.com/hibiken/asynq/tools | ||||
|  | ||||
| go 1.18 | ||||
| go 1.22 | ||||
|  | ||||
| require ( | ||||
| 	github.com/MakeNowJust/heredoc/v2 v2.0.1 | ||||
| 	github.com/fatih/color v1.9.0 | ||||
| 	github.com/fatih/color v1.18.0 | ||||
| 	github.com/gdamore/tcell/v2 v2.5.1 | ||||
| 	github.com/go-redis/redis/v8 v8.11.4 | ||||
| 	github.com/google/go-cmp v0.5.6 | ||||
| 	github.com/hibiken/asynq v0.23.0 | ||||
| 	github.com/google/go-cmp v0.6.0 | ||||
| 	github.com/hibiken/asynq v0.25.0 | ||||
| 	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.16 | ||||
| 	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.7.0 | ||||
| 	github.com/spf13/cobra v1.1.1 | ||||
| 	github.com/spf13/pflag v1.0.5 | ||||
| 	github.com/spf13/viper v1.7.0 | ||||
| @@ -25,14 +25,14 @@ require ( | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/fsnotify/fsnotify v1.4.9 // indirect | ||||
| 	github.com/gdamore/encoding v1.0.0 // indirect | ||||
| 	github.com/golang/protobuf v1.5.2 // indirect | ||||
| 	github.com/google/uuid v1.3.0 // indirect | ||||
| 	github.com/golang/protobuf v1.5.3 // indirect | ||||
| 	github.com/google/uuid v1.6.0 // indirect | ||||
| 	github.com/hashicorp/hcl v1.0.0 // indirect | ||||
| 	github.com/inconshreveable/mousetrap v1.0.0 // indirect | ||||
| 	github.com/lucasb-eyer/go-colorful v1.2.0 // indirect | ||||
| 	github.com/magiconair/properties v1.8.1 // indirect | ||||
| 	github.com/mattn/go-colorable v0.1.4 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.11 // indirect | ||||
| 	github.com/mattn/go-colorable v0.1.13 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.20 // indirect | ||||
| 	github.com/matttproud/golang_protobuf_extensions v1.0.1 // indirect | ||||
| 	github.com/mitchellh/mapstructure v1.1.2 // indirect | ||||
| 	github.com/pelletier/go-toml v1.2.0 // indirect | ||||
| @@ -42,14 +42,14 @@ require ( | ||||
| 	github.com/rivo/uniseg v0.2.0 // indirect | ||||
| 	github.com/robfig/cron/v3 v3.0.1 // indirect | ||||
| 	github.com/spf13/afero v1.1.2 // indirect | ||||
| 	github.com/spf13/cast v1.3.1 // indirect | ||||
| 	github.com/spf13/cast v1.7.0 // indirect | ||||
| 	github.com/spf13/jwalterweatherman v1.0.0 // indirect | ||||
| 	github.com/subosito/gotenv v1.2.0 // indirect | ||||
| 	golang.org/x/sys v0.0.0-20220318055525-2edf467146b5 // indirect | ||||
| 	golang.org/x/sys v0.26.0 // indirect | ||||
| 	golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf // indirect | ||||
| 	golang.org/x/text v0.3.7 // indirect | ||||
| 	golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 // indirect | ||||
| 	google.golang.org/protobuf v1.26.0 // indirect | ||||
| 	golang.org/x/text v0.3.8 // indirect | ||||
| 	golang.org/x/time v0.7.0 // indirect | ||||
| 	google.golang.org/protobuf v1.35.1 // indirect | ||||
| 	gopkg.in/ini.v1 v1.51.0 // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										78
									
								
								tools/go.sum
									
									
									
									
									
								
							
							
						
						
									
										78
									
								
								tools/go.sum
									
									
									
									
									
								
							| @@ -31,6 +31,10 @@ 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/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= | ||||
| github.com/bketelsen/crypt v0.0.3-0.20200106085610-5cbc8cc4026c/go.mod h1:MKsuJmJgSg28kpZDP6UIiPt0e0Oz0kqKNGyRaWEPv84= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| @@ -54,8 +58,10 @@ github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8 | ||||
| github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= | ||||
| github.com/fatih/color v1.9.0 h1:8xPHl4/q1VyqGIPif1F+1V3Y3lSmrq01EabUW3CoW5s= | ||||
| github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= | ||||
| github.com/fatih/color v1.18.0 h1:S8gINlzdQ840/4pfAwic/ZE0djQEH3wM94VfqLTZcOM= | ||||
| github.com/fatih/color v1.18.0/go.mod h1:4FelSpRwEGDpQ12mAdzqdOukCy4u8WUtOY6lkT/6HfU= | ||||
| github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||
| github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= | ||||
| 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= | ||||
| @@ -72,7 +78,6 @@ github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9 | ||||
| github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | ||||
| github.com/go-redis/redis/v8 v8.11.2/go.mod h1:DLomh7y2e3ggQXQLd1YgmvIfecPJoFl7WU5SOQ/r06M= | ||||
| github.com/go-redis/redis/v8 v8.11.4 h1:kHoYkfZP6+pe04aFTnhDH6GDROa5yJdHJVNxV3F46Tg= | ||||
| github.com/go-redis/redis/v8 v8.11.4/go.mod h1:2Z2wHZXdQpCDXEGzqMockDpNyYvi2l4Pxt6RJr792+w= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| @@ -95,8 +100,9 @@ github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QD | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= | ||||
| github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
| @@ -106,16 +112,18 @@ github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.6 h1:BKbKCqvP6I+rmFHt06ZmyQtvB8xAkWdhFyr0ZUNZcxQ= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= | ||||
| github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||
| github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= | ||||
| github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= | ||||
| 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.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= | ||||
| github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= | ||||
| github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8= | ||||
| @@ -146,8 +154,8 @@ github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0m | ||||
| github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= | ||||
| github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= | ||||
| github.com/hibiken/asynq v0.19.0/go.mod h1:tyc63ojaW8SJ5SBm8mvI4DDONsguP5HE85EEl4Qr5Ig= | ||||
| github.com/hibiken/asynq v0.23.0 h1:kmKkNFgqiXBatC8oz94Mer6uvKoGn4STlIVDV5wnKyE= | ||||
| github.com/hibiken/asynq v0.23.0/go.mod h1:K70jPVx+CAmmQrXot7Dru0D52EO7ob4BIun3ri5z1Qw= | ||||
| github.com/hibiken/asynq v0.25.0 h1:VCPyRRrrjFChsTSI8x5OCPu51MlEz6Rk+1p0kHKnZug= | ||||
| github.com/hibiken/asynq v0.25.0/go.mod h1:DYQ1etBEl2Y+uSkqFElGYbk3M0ujLVwCfWE+TlvxtEk= | ||||
| github.com/hibiken/asynq/x v0.0.0-20220131170841-349f4c50fb1d h1:Er+U+9PmnyRHRDQjSjRQ24HoWvOY7w9Pk7bUPYM3Ags= | ||||
| github.com/hibiken/asynq/x v0.0.0-20220131170841-349f4c50fb1d/go.mod h1:VmxwMfMKyb6gyv8xG0oOBMXIhquWKPx+zPtbVBd2Q1s= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| @@ -168,24 +176,27 @@ github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+o | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/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/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= | ||||
| 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/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY= | ||||
| github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0= | ||||
| github.com/magiconair/properties v1.8.1 h1:ZC2Vc7/ZFkGmsVC9KvOjumD+G5lXy2RtTKyzRKO2BQ4= | ||||
| github.com/magiconair/properties v1.8.1/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | ||||
| github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= | ||||
| github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA= | ||||
| github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= | ||||
| github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= | ||||
| github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= | ||||
| github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||
| github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= | ||||
| github.com/mattn/go-isatty v0.0.11 h1:FxPOTFNqGkuDUGi3H/qkUbQO4ZiBa2brKq5r0l8TGeM= | ||||
| github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= | ||||
| github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU= | ||||
| github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= | ||||
| github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= | ||||
| github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | ||||
| github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= | ||||
| github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc= | ||||
| github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= | ||||
| 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/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= | ||||
| @@ -206,18 +217,15 @@ github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3Rllmb | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= | ||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= | ||||
| github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U= | ||||
| 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/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= | ||||
| github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc= | ||||
| @@ -232,8 +240,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 v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.11.0 h1:HNkLOAEQMIDv/K+04rukrLx6ch7msSRwf3/SASFAGtQ= | ||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_golang v1.11.1 h1:+4eQaD7vAZ6DsfsxB15hbE0odUjGI5ARs9yskGu1v4s= | ||||
| github.com/prometheus/client_golang v1.11.1/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| @@ -252,12 +261,16 @@ github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4O | ||||
| 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/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU= | ||||
| github.com/redis/go-redis/v9 v9.7.0 h1:HhLSs+B6O021gwzl+locl0zEDnyNkxMtf/Z3NNBMa9E= | ||||
| github.com/redis/go-redis/v9 v9.7.0/go.mod h1:f6zhXITC7JUJIlPEiBOTXxJgPLdZcA93GewI7inzyWw= | ||||
| github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= | ||||
| github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= | ||||
| github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs= | ||||
| github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro= | ||||
| github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= | ||||
| github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | ||||
| github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= | ||||
| github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= | ||||
| github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= | ||||
| github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= | ||||
| github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= | ||||
| @@ -274,8 +287,9 @@ github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasO | ||||
| github.com/spf13/afero v1.1.2 h1:m8/z1t7/fwjysjQRYbP0RD+bUIF/8tJwPdEZsI83ACI= | ||||
| github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= | ||||
| github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | ||||
| 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.7.0 h1:ntdiHjuueXFgm5nzDRdOS4yfT43P5Fnud6DH50rz/7w= | ||||
| github.com/spf13/cast v1.7.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= | ||||
| github.com/spf13/cobra v1.1.1 h1:KfztREH0tPxJJ+geloSLaAkaPkr4ki2Er5quFV1TDo4= | ||||
| github.com/spf13/cobra v1.1.1/go.mod h1:WnodtKOvamDL/PwE2M4iKs8aMDBZ5Q5klgD3qfVJQMI= | ||||
| github.com/spf13/jwalterweatherman v1.0.0 h1:XHEdyB+EcvlqZamSM4ZOMGlc93t6AcsBEu9Gc1vn7yk= | ||||
| @@ -302,8 +316,9 @@ go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= | ||||
| go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= | ||||
| go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= | ||||
| go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= | ||||
| go.uber.org/goleak v0.10.0 h1:G3eWbSNIskeRqtsN/1uI5B+eP73y3JUuBsv9AZjehb4= | ||||
| go.uber.org/goleak v0.10.0/go.mod h1:VCZuO8V8mFPlL0F5J5GK1rtHV3DrFcQ1R8ryq7FK0aI= | ||||
| go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||
| go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||
| go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= | ||||
| go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| @@ -353,7 +368,6 @@ golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/ | ||||
| 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-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= | ||||
| @@ -374,7 +388,6 @@ golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5h | ||||
| golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/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-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190312061237-fead79001313/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= | ||||
| @@ -384,7 +397,6 @@ golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7w | ||||
| golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/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-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -396,8 +408,11 @@ golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7w | ||||
| 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-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220318055525-2edf467146b5 h1:saXMvIOKvRFwbOMicHXr0B1uwoxq9dGmLe5ExMES6c4= | ||||
| golang.org/x/sys v0.0.0-20220318055525-2edf467146b5/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo= | ||||
| golang.org/x/sys v0.26.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-20201210144234-2321bbc49cbf h1:MZ2shdL+ZM/XzY3ZGOnh4Nlpnxz5GSOhOmtHo3iPU6M= | ||||
| golang.org/x/term v0.0.0-20201210144234-2321bbc49cbf/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| @@ -406,11 +421,13 @@ golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3 | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.3.8 h1:nAL+RVCQ9uMn3vJZbV+MRnydTJFPf8qqY42YiA6MrqY= | ||||
| golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= | ||||
| golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.7.0 h1:ntUhktv3OPE6TgYxXWv9vKvUSJyIFJlyohwbkEwPrKQ= | ||||
| golang.org/x/time v0.7.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/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= | ||||
| @@ -434,7 +451,6 @@ golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4f | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= | ||||
| google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= | ||||
| @@ -470,8 +486,9 @@ google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2 | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.35.1 h1:m3LfL6/Ca+fqnjnlqQXNpFPABW1UD7mjh8KO2mKFytA= | ||||
| google.golang.org/protobuf v1.35.1/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| @@ -482,7 +499,6 @@ gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMy | ||||
| gopkg.in/ini.v1 v1.51.0 h1:AQvPpx3LzTDM0AjnIRlVFwFFGC+npRopjZxLJj6gdno= | ||||
| gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= | ||||
| 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.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
|   | ||||
							
								
								
									
										25
									
								
								x/go.mod
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								x/go.mod
									
									
									
									
									
								
							| @@ -1,10 +1,25 @@ | ||||
| module github.com/hibiken/asynq/x | ||||
|  | ||||
| go 1.16 | ||||
| go 1.22 | ||||
|  | ||||
| require ( | ||||
| 	github.com/go-redis/redis/v8 v8.11.4 | ||||
| 	github.com/google/uuid v1.3.0 | ||||
| 	github.com/hibiken/asynq v0.21.0 | ||||
| 	github.com/prometheus/client_golang v1.11.0 | ||||
| 	github.com/google/uuid v1.6.0 | ||||
| 	github.com/hibiken/asynq v0.25.0 | ||||
| 	github.com/prometheus/client_golang v1.20.5 | ||||
| 	github.com/redis/go-redis/v9 v9.7.0 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/beorn7/perks v1.0.1 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.3.0 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect | ||||
| 	github.com/prometheus/client_model v0.6.1 // indirect | ||||
| 	github.com/prometheus/common v0.55.0 // indirect | ||||
| 	github.com/prometheus/procfs v0.15.1 // indirect | ||||
| 	github.com/robfig/cron/v3 v3.0.1 // indirect | ||||
| 	github.com/spf13/cast v1.7.0 // indirect | ||||
| 	golang.org/x/sys v0.26.0 // indirect | ||||
| 	golang.org/x/time v0.7.0 // indirect | ||||
| 	google.golang.org/protobuf v1.35.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										296
									
								
								x/go.sum
									
									
									
									
									
								
							
							
						
						
									
										296
									
								
								x/go.sum
									
									
									
									
									
								
							| @@ -1,258 +1,50 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= | ||||
| github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= | ||||
| github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= | ||||
| github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= | ||||
| github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | ||||
| github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= | ||||
| github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= | ||||
| github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= | ||||
| github.com/bsm/gomega v1.27.10/go.mod h1:JyEr/xRbxbtgWNi8tIEVPUYZ5Dzef52k01W3YH0H+O0= | ||||
| github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= | ||||
| github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| 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/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | ||||
| github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
| github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= | ||||
| github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= | ||||
| github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | ||||
| github.com/go-redis/redis/v8 v8.11.2/go.mod h1:DLomh7y2e3ggQXQLd1YgmvIfecPJoFl7WU5SOQ/r06M= | ||||
| github.com/go-redis/redis/v8 v8.11.4 h1:kHoYkfZP6+pe04aFTnhDH6GDROa5yJdHJVNxV3F46Tg= | ||||
| github.com/go-redis/redis/v8 v8.11.4/go.mod h1:2Z2wHZXdQpCDXEGzqMockDpNyYvi2l4Pxt6RJr792+w= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.5.6 h1:BKbKCqvP6I+rmFHt06ZmyQtvB8xAkWdhFyr0ZUNZcxQ= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/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.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= | ||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hibiken/asynq v0.21.0 h1:uH9XogJhjq/S39E0/DEPWLZQ6hHJ73UiblZTe4RzHwA= | ||||
| github.com/hibiken/asynq v0.21.0/go.mod h1:tyc63ojaW8SJ5SBm8mvI4DDONsguP5HE85EEl4Qr5Ig= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= | ||||
| github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||
| github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= | ||||
| github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/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/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= | ||||
| github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/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-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= | ||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.15.0/go.mod h1:hF8qUzuuC8DJGygJH3726JnCZX4MYbRB8yFfISqnKUg= | ||||
| github.com/onsi/ginkgo v1.16.4 h1:29JGrr5oVBm5ulCWet69zQkzWipVXIol6ygQUe/EzNc= | ||||
| github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= | ||||
| github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= | ||||
| github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= | ||||
| github.com/onsi/gomega v1.10.5/go.mod h1:gza4q3jKQJijlu05nKWRCW/GavJumGt8aNRxWg7mt48= | ||||
| github.com/onsi/gomega v1.16.0 h1:6gjqkI8iiRHMvdccRJM8rVKjCWk6ZIm6FTm3ddIe4/c= | ||||
| github.com/onsi/gomega v1.16.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= | ||||
| github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= | ||||
| github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.11.0 h1:HNkLOAEQMIDv/K+04rukrLx6ch7msSRwf3/SASFAGtQ= | ||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= | ||||
| github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= | ||||
| github.com/prometheus/common v0.26.0 h1:iMAkS2TDoNWnKM+Kopnx/8tnEStIfpYA0ur0xQzzhMQ= | ||||
| github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= | ||||
| github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= | ||||
| github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= | ||||
| github.com/prometheus/procfs v0.6.0 h1:mxy4L2jP6qMonqmq+aTtOx1ifVWUgG/TAmntgbh3xv4= | ||||
| github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||
| github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||
| github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hibiken/asynq v0.25.0 h1:VCPyRRrrjFChsTSI8x5OCPu51MlEz6Rk+1p0kHKnZug= | ||||
| github.com/hibiken/asynq v0.25.0/go.mod h1:DYQ1etBEl2Y+uSkqFElGYbk3M0ujLVwCfWE+TlvxtEk= | ||||
| github.com/klauspost/compress v1.17.9 h1:6KIumPrER1LHsvBVuDa0r5xaG0Es51mhhB9BQB2qeMA= | ||||
| github.com/klauspost/compress v1.17.9/go.mod h1:Di0epgTjJY877eYKx5yC51cX2A2Vl2ibi7bDH9ttBbw= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA= | ||||
| github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= | ||||
| github.com/prometheus/client_golang v1.20.5 h1:cxppBPuYhUnsO6yo/aoRol4L7q7UFfdm+bR9r+8l63Y= | ||||
| github.com/prometheus/client_golang v1.20.5/go.mod h1:PIEt8X02hGcP8JWbeHyeZ53Y/jReSnHgO035n//V5WE= | ||||
| github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E= | ||||
| github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY= | ||||
| github.com/prometheus/common v0.55.0 h1:KEi6DK7lXW/m7Ig5i47x0vRzuBsHuvJdi5ee6Y3G1dc= | ||||
| github.com/prometheus/common v0.55.0/go.mod h1:2SECS4xJG1kd8XF9IcM1gMX6510RAEL65zxzNImwdc8= | ||||
| github.com/prometheus/procfs v0.15.1 h1:YagwOFzUgYfKKHX6Dr+sHT7km/hxC76UB0learggepc= | ||||
| github.com/prometheus/procfs v0.15.1/go.mod h1:fB45yRUv8NstnjriLhBQLuOUt+WW4BsoGhij/e3PBqk= | ||||
| github.com/redis/go-redis/v9 v9.7.0 h1:HhLSs+B6O021gwzl+locl0zEDnyNkxMtf/Z3NNBMa9E= | ||||
| github.com/redis/go-redis/v9 v9.7.0/go.mod h1:f6zhXITC7JUJIlPEiBOTXxJgPLdZcA93GewI7inzyWw= | ||||
| 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/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||
| github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= | ||||
| github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= | ||||
| github.com/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= | ||||
| github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| go.uber.org/goleak v0.10.0 h1:G3eWbSNIskeRqtsN/1uI5B+eP73y3JUuBsv9AZjehb4= | ||||
| go.uber.org/goleak v0.10.0/go.mod h1:VCZuO8V8mFPlL0F5J5GK1rtHV3DrFcQ1R8ryq7FK0aI= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781 h1:DzZ89McO9/gWPsQXS/FVKAlG02ZjaQ6AlZRBimEYOd0= | ||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40 h1:JWgyZ1qgdTaF3N3oxC+MdTV7qvEEgHo3otj+HB5CM7Q= | ||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6 h1:aRYxNxv6iGQlyVaZmk6ZgYEDa+Jg18DxebPSrd6bg1M= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ= | ||||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= | ||||
| golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= | ||||
| google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | ||||
| google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | ||||
| google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ= | ||||
| github.com/rogpeppe/go-internal v1.10.0/go.mod h1:UQnix2H7Ngw/k4C5ijL5+65zddjncjaFoBhdsK/akog= | ||||
| github.com/spf13/cast v1.7.0 h1:ntdiHjuueXFgm5nzDRdOS4yfT43P5Fnud6DH50rz/7w= | ||||
| github.com/spf13/cast v1.7.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= | ||||
| go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||
| go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||
| golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo= | ||||
| golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/time v0.7.0 h1:ntUhktv3OPE6TgYxXWv9vKvUSJyIFJlyohwbkEwPrKQ= | ||||
| golang.org/x/time v0.7.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| google.golang.org/protobuf v1.35.1 h1:m3LfL6/Ca+fqnjnlqQXNpFPABW1UD7mjh8KO2mKFytA= | ||||
| google.golang.org/protobuf v1.35.1/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= | ||||
|   | ||||
| @@ -7,9 +7,9 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| // NewSemaphore creates a counting Semaphore for the given scope with the given number of tokens. | ||||
| @@ -110,5 +110,5 @@ func (s *Semaphore) Close() error { | ||||
| } | ||||
|  | ||||
| func semaphoreKey(scope string) string { | ||||
| 	return fmt.Sprintf("asynq:sema:%s", scope) | ||||
| 	return "asynq:sema:" + scope | ||||
| } | ||||
|   | ||||
| @@ -8,11 +8,11 @@ import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| 	"github.com/redis/go-redis/v9" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| @@ -91,7 +91,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | ||||
| 			maxConcurrency: 3, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-1", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -104,7 +104,7 @@ func TestNewSemaphore_Acquire(t *testing.T) { | ||||
| 			maxConcurrency: 3, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString(), uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-2", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -211,7 +211,7 @@ func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | ||||
|  | ||||
| 	// adding a set member to mimic the case where token is acquired but the goroutine crashed, | ||||
| 	// in which case, the token will not be explicitly removed and should be present already | ||||
| 	rc.ZAdd(context.Background(), semaphoreKey("stale-token"), &redis.Z{ | ||||
| 	rc.ZAdd(context.Background(), semaphoreKey("stale-token"), redis.Z{ | ||||
| 		Score:  float64(time.Now().Add(-10 * time.Second).Unix()), | ||||
| 		Member: taskID, | ||||
| 	}) | ||||
| @@ -219,7 +219,7 @@ func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | ||||
| 	sema := NewSemaphore(opt, "stale-token", 1) | ||||
| 	defer sema.Close() | ||||
|  | ||||
| 	ctx, cancel := asynqcontext.New(&base.TaskMessage{ | ||||
| 	ctx, cancel := asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 		ID:    taskID, | ||||
| 		Queue: "task-1", | ||||
| 	}, time.Now().Add(time.Second)) | ||||
| @@ -248,7 +248,7 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 			name:    "task-5", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-3", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -259,7 +259,7 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 			name:    "task-6", | ||||
| 			taskIDs: []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -277,9 +277,9 @@ func TestNewSemaphore_Release(t *testing.T) { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []*redis.Z | ||||
| 			var members []redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, &redis.Z{ | ||||
| 				members = append(members, redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
| @@ -337,7 +337,7 @@ func TestNewSemaphore_Release_Error(t *testing.T) { | ||||
| 			name:    "task-8", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 				return asynqcontext.New(context.Background(), &base.TaskMessage{ | ||||
| 					ID:    testID, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| @@ -356,9 +356,9 @@ func TestNewSemaphore_Release_Error(t *testing.T) { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []*redis.Z | ||||
| 			var members []redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, &redis.Z{ | ||||
| 				members = append(members, redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user