mirror of
				https://github.com/hibiken/asynq.git
				synced 2025-10-21 21:46:12 +08:00 
			
		
		
		
	Compare commits
	
		
			139 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 413afc2ab6 | ||
|  | 6bb4818509 | ||
|  | f4ddac4dcc | ||
|  | 4638405cbd | ||
|  | 9e2f88c00d | ||
|  | dbdd9c6d5f | ||
|  | 2261c7c9a0 | ||
|  | 83cae4bb24 | ||
|  | 23c522dc9f | ||
|  | 0d2c0f612b | ||
|  | d612a8a9e4 | ||
|  | b3ef9e91a9 | ||
|  | 05534c6f24 | ||
|  | f0db219f6a | ||
|  | 3ae0e7f528 | ||
|  | 421dc584ff | ||
|  | cfd1a1dfe8 | ||
|  | c197902dc0 | ||
|  | e6355bf3f5 | ||
|  | 95c90a5cb8 | ||
|  | 6817af366a | ||
|  | 4bce28d677 | ||
|  | 73f930313c | ||
|  | bff2a05d59 | ||
|  | 684a7e0c98 | ||
|  | 46b23d6495 | ||
|  | c0ae62499f | ||
|  | 7744ade362 | ||
|  | f532c95394 | ||
|  | ff6768f9bb | ||
|  | d5e9f3b1bd | ||
|  | d02b722d8a | ||
|  | 99c7ebeef2 | ||
|  | bf54621196 | ||
|  | 27baf6de0d | ||
|  | 1bd0bee1e5 | ||
|  | a9feec5967 | ||
|  | e01c6379c8 | ||
|  | a0df047f71 | ||
|  | 68dd6d9a9d | ||
|  | 6cce31a134 | ||
|  | f9d7af3def | ||
|  | b0321fb465 | ||
|  | 7776c7ae53 | ||
|  | 709ca79a2b | ||
|  | 08d8f0b37c | ||
|  | 385323b679 | ||
|  | 77604af265 | ||
|  | 4765742e8a | ||
|  | 68839dc9d3 | ||
|  | 8922d2423a | ||
|  | b358de907e | ||
|  | 8ee1825e67 | ||
|  | c8bda26bed | ||
|  | 8aeeb61c9d | ||
|  | 96c51fdc23 | ||
|  | ea9086fd8b | ||
|  | e63d51da0c | ||
|  | cd351d49b9 | ||
|  | 87264b66f3 | ||
|  | 62168b8d0d | ||
|  | 840f7245b1 | ||
|  | 12f4c7cf6e | ||
|  | 0ec3b55e6b | ||
|  | 4bcc5ab6aa | ||
|  | 456edb6b71 | ||
|  | b835090ad8 | ||
|  | 09cbea66f6 | ||
|  | b9c2572203 | ||
|  | 0bf767cf21 | ||
|  | 1812d05d21 | ||
|  | 4af65d5fa5 | ||
|  | a19ad19382 | ||
|  | 8117ce8972 | ||
|  | d98ecdebb4 | ||
|  | ffe9aa74b3 | ||
|  | d2d4029aba | ||
|  | 76bd865ebc | ||
|  | 136d1c9ea9 | ||
|  | 52e04355d3 | ||
|  | cde3e57c6c | ||
|  | dd66acef1b | ||
|  | 30a3d9641a | ||
|  | 961582cba6 | ||
|  | 430dbb298e | ||
|  | 675826be5f | ||
|  | 62f4e46b73 | ||
|  | a500f8a534 | ||
|  | bcfeff38ed | ||
|  | 12a90f6a8d | ||
|  | 807624e7dd | ||
|  | 4d65024bd7 | ||
|  | 76486b5cb4 | ||
|  | 1db516c53c | ||
|  | cb5bdf245c | ||
|  | 267493ccef | ||
|  | 5d7f1b6a80 | ||
|  | 77ded502ab | ||
|  | f2284be43d | ||
|  | 3cadab55cb | ||
|  | 298a420f9f | ||
|  | b1d717c842 | ||
|  | 56e5762eea | ||
|  | 5ec41e388b | ||
|  | 9c95c41651 | ||
|  | 476812475e | ||
|  | 7af3981929 | ||
|  | 2516c4baba | ||
|  | ebe482a65c | ||
|  | 3e9fc2f972 | ||
|  | 63ce9ed0f9 | ||
|  | 32d3f329b9 | ||
|  | 544c301a8b | ||
|  | 8b997d2fab | ||
|  | 901105a8d7 | ||
|  | aaa3f1d4fd | ||
|  | 4722ca2d3d | ||
|  | 6a9d9fd717 | ||
|  | de28c1ea19 | ||
|  | f618f5b1f5 | ||
|  | aa936466b3 | ||
|  | 5d1ec70544 | ||
|  | d1d3be9b00 | ||
|  | bc77f6fe14 | ||
|  | efe197a47b | ||
|  | 97b5516183 | ||
|  | 8eafa03ca7 | ||
|  | 430b01c9aa | ||
|  | 14c381dc40 | ||
|  | e13122723a | ||
|  | eba7c4e085 | ||
|  | bfde0b6283 | ||
|  | afde6a7266 | ||
|  | 6529a1e0b1 | ||
|  | c9a6ab8ae1 | ||
|  | 557c1a5044 | ||
|  | 0236eb9a1c | ||
|  | 3c2b2cf4a3 | ||
|  | 04df71198d | 
							
								
								
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| # 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'] | ||||
							
								
								
									
										82
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								.github/workflows/benchstat.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| # This workflow runs benchmarks against the current branch, | ||||
| # compares them to benchmarks against master, | ||||
| # and uploads the results as an artifact. | ||||
|  | ||||
| name: benchstat | ||||
|  | ||||
| on: [pull_request] | ||||
|  | ||||
| jobs: | ||||
|   incoming: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.16.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a new.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|           path: new.txt | ||||
|  | ||||
|   current: | ||||
|     runs-on: ubuntu-latest | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           ref: master | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.15.x | ||||
|       - name: Benchmark | ||||
|         run: go test -run=^$ -bench=. -count=5 -timeout=60m ./... | tee -a old.txt | ||||
|       - name: Upload Benchmark | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: bench-current | ||||
|           path: old.txt | ||||
|  | ||||
|   benchstat: | ||||
|     needs: [incoming, current] | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: 1.15.x | ||||
|       - name: Install benchstat | ||||
|         run: go get -u golang.org/x/perf/cmd/benchstat | ||||
|       - name: Download Incoming | ||||
|         uses: actions/download-artifact@v2 | ||||
|         with: | ||||
|           name: bench-incoming | ||||
|       - name: Download Current | ||||
|         uses: actions/download-artifact@v2 | ||||
|         with: | ||||
|           name: bench-current | ||||
|       - name: Benchstat Results | ||||
|         run: benchstat old.txt new.txt | tee -a benchstat.txt | ||||
|       - name: Upload benchstat results | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: benchstat | ||||
|           path: benchstat.txt | ||||
							
								
								
									
										35
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| name: build | ||||
|  | ||||
| on: [push, pull_request] | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         go-version: [1.14.x, 1.15.x, 1.16.x, 1.17.x] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     services: | ||||
|       redis: | ||||
|         image: redis | ||||
|         ports: | ||||
|           - 6379:6379 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: ${{ matrix.go-version }} | ||||
|  | ||||
|       - name: Build | ||||
|         run: go build -v ./... | ||||
|  | ||||
|       - name: Test | ||||
|         run: go test -race -v -coverprofile=coverage.txt -covermode=atomic ./... | ||||
|  | ||||
|       - name: Benchmark Test | ||||
|         run: go test -run=^$ -bench=. -loglevel=debug ./... | ||||
|  | ||||
|       - name: Upload coverage to Codecov | ||||
|         uses: codecov/codecov-action@v1 | ||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -19,3 +19,7 @@ | ||||
|  | ||||
| # Ignore asynq config file | ||||
| .asynq.* | ||||
|  | ||||
| # Ignore editor config files | ||||
| .vscode | ||||
| .idea | ||||
							
								
								
									
										13
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,13 +0,0 @@ | ||||
| language: go | ||||
| go_import_path: github.com/hibiken/asynq | ||||
| git: | ||||
|   depth: 1 | ||||
| go: [1.13.x, 1.14.x, 1.15.x] | ||||
| script: | ||||
|   - go test -race -v -coverprofile=coverage.txt -covermode=atomic ./... | ||||
|   - go test -run=^$ -bench=. -loglevel=debug ./... | ||||
| services: | ||||
|   - redis-server | ||||
| after_success: | ||||
|   - travis_wait 60 bash ./.travis/benchstat.sh | ||||
|   - bash <(curl -s https://codecov.io/bash) | ||||
| @@ -1,20 +0,0 @@ | ||||
| if [ "${TRAVIS_PULL_REQUEST_BRANCH:-$TRAVIS_BRANCH}" != "master" ]; then | ||||
|     REMOTE_URL="$(git config --get remote.origin.url)"; | ||||
|     cd ${TRAVIS_BUILD_DIR}/.. && \ | ||||
|     git clone ${REMOTE_URL} "${TRAVIS_REPO_SLUG}-bench" && \ | ||||
|     # turn the detached message off | ||||
|     git config --global advice.detachedHead false && \ | ||||
|     cd "${TRAVIS_REPO_SLUG}-bench" && \ | ||||
|  | ||||
|     # Benchmark master | ||||
|     git checkout master && \ | ||||
|     go test -run=^$ -bench=. -count=5 -timeout=60m -benchmem ./... > master.txt && \ | ||||
|  | ||||
|     # Benchmark feature branch | ||||
|     git checkout ${TRAVIS_COMMIT} && \ | ||||
|     go test -run=^$ -bench=. -count=5 -timeout=60m -benchmem ./... > feature.txt && \ | ||||
|  | ||||
|     # compare two benchmarks | ||||
|     go get -u golang.org/x/perf/cmd/benchstat && \ | ||||
|     benchstat master.txt feature.txt; | ||||
| fi | ||||
							
								
								
									
										130
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										130
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -7,6 +7,136 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 | ||||
|  | ||||
| ## [Unreleased] | ||||
|  | ||||
| ## [0.19.0] - 2021-11-06 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - `NewTask` takes `Option` as variadic argument | ||||
| - Bumped minimum supported go version to 1.14 (i.e. go1.14 or higher is required). | ||||
|  | ||||
| ### Added | ||||
|  | ||||
| - `Retention` option is added to allow user to specify task retention duration after completion. | ||||
| - `TaskID` option is added to allow user to specify task ID. | ||||
| - `ErrTaskIDConflict` sentinel error value is added. | ||||
| - `ResultWriter` type is added and provided through `Task.ResultWriter` method. | ||||
| - `TaskInfo` has new fields `CompletedAt`, `Result` and `Retention`. | ||||
|  | ||||
| ### Removed | ||||
|  | ||||
| - `Client.SetDefaultOptions` is removed. Use `NewTask` instead to pass default options for tasks. | ||||
|  | ||||
| ## [0.18.6] - 2021-10-03 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - Updated `github.com/go-redis/redis` package to v8 | ||||
|  | ||||
| ## [0.18.5] - 2021-09-01 | ||||
|  | ||||
| ### Added | ||||
|  | ||||
| - `IsFailure` config option is added to determine whether error returned from Handler counts as a failure. | ||||
|  | ||||
| ## [0.18.4] - 2021-08-17 | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Scheduler methods are now thread-safe. It's now safe to call `Register` and `Unregister` concurrently. | ||||
|  | ||||
| ## [0.18.3] - 2021-08-09 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - `Client.Enqueue` no longer enqueues tasks with empty typename; Error message is returned. | ||||
|  | ||||
| ## [0.18.2] - 2021-07-15 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - Changed `Queue` function to not to convert the provided queue name to lowercase. Queue names are now case-sensitive. | ||||
| - `QueueInfo.MemoryUsage` is now an approximate usage value. | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Fixed latency issue around memory usage (see https://github.com/hibiken/asynq/issues/309). | ||||
|  | ||||
| ## [0.18.1] - 2021-07-04 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - Changed to execute task recovering logic when server starts up; Previously it needed to wait for a minute for task recovering logic to exeucte. | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Fixed task recovering logic to execute every minute | ||||
|  | ||||
| ## [0.18.0] - 2021-06-29 | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - NewTask function now takes array of bytes as payload. | ||||
| - Task `Type` and `Payload` should be accessed by a method call. | ||||
| - `Server` API has changed. Renamed `Quiet` to `Stop`. Renamed `Stop` to `Shutdown`. _Note:_ As a result of this renaming, the behavior of `Stop` has changed. Please update the exising code to call `Shutdown` where it used to call `Stop`. | ||||
| - `Scheduler` API has changed. Renamed `Stop` to `Shutdown`. | ||||
| - Requires redis v4.0+ for multiple field/value pair support | ||||
| - `Client.Enqueue` now returns `TaskInfo` | ||||
| - `Inspector.RunTaskByKey` is replaced with `Inspector.RunTask` | ||||
| - `Inspector.DeleteTaskByKey` is replaced with `Inspector.DeleteTask` | ||||
| - `Inspector.ArchiveTaskByKey` is replaced with `Inspector.ArchiveTask` | ||||
| - `inspeq` package is removed. All types and functions from the package is moved to `asynq` package. | ||||
| - `WorkerInfo` field names have changed. | ||||
| - `Inspector.CancelActiveTask` is renamed to `Inspector.CancelProcessing` | ||||
|  | ||||
| ## [0.17.2] - 2021-06-06 | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Free unique lock when task is deleted (https://github.com/hibiken/asynq/issues/275). | ||||
|  | ||||
| ## [0.17.1] - 2021-04-04 | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Fix bug in internal `RDB.memoryUsage` method. | ||||
|  | ||||
| ## [0.17.0] - 2021-03-24 | ||||
|  | ||||
| ### Added | ||||
|  | ||||
| - `DialTimeout`, `ReadTimeout`, and `WriteTimeout` options are added to `RedisConnOpt`. | ||||
|  | ||||
| ## [0.16.1] - 2021-03-20 | ||||
|  | ||||
| ### Fixed | ||||
|  | ||||
| - Replace `KEYS` command with `SCAN` as recommended by [redis doc](https://redis.io/commands/KEYS). | ||||
|  | ||||
| ## [0.16.0] - 2021-03-10 | ||||
|  | ||||
| ### Added | ||||
|  | ||||
| - `Unregister` method is added to `Scheduler` to remove a registered entry. | ||||
|  | ||||
| ## [0.15.0] - 2021-01-31 | ||||
|  | ||||
| **IMPORTATNT**: All `Inspector` related code are moved to subpackage "github.com/hibiken/asynq/inspeq" | ||||
|  | ||||
| ### Changed | ||||
|  | ||||
| - `Inspector` related code are moved to subpackage "github.com/hibken/asynq/inspeq". | ||||
| - `RedisConnOpt` interface has changed slightly. If you have been passing `RedisClientOpt`, `RedisFailoverClientOpt`, or `RedisClusterClientOpt` as a pointer, | ||||
|   update your code to pass as a value. | ||||
| - `ErrorMsg` field in `RetryTask` and `ArchivedTask` was renamed to `LastError`. | ||||
|  | ||||
| ### Added | ||||
|  | ||||
| - `MaxRetry`, `Retried`, `LastError` fields were added to all task types returned from `Inspector`. | ||||
| - `MemoryUsage` field was added to `QueueStats`. | ||||
| - `DeleteAllPendingTasks`, `ArchiveAllPendingTasks` were added to `Inspector` | ||||
| - `DeleteTaskByKey` and `ArchiveTaskByKey` now supports deleting/archiving `PendingTask`. | ||||
| - asynq CLI now supports deleting/archiving pending tasks. | ||||
|  | ||||
| ## [0.14.1] - 2021-01-19 | ||||
|  | ||||
| ### Fixed | ||||
|   | ||||
							
								
								
									
										7
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| ROOT_DIR:=$(shell dirname $(realpath $(firstword $(MAKEFILE_LIST)))) | ||||
|  | ||||
| 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 | ||||
							
								
								
									
										216
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										216
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,41 +1,35 @@ | ||||
| # Asynq | ||||
| <img src="https://user-images.githubusercontent.com/11155743/114697792-ffbfa580-9d26-11eb-8e5b-33bef69476dc.png" alt="Asynq logo" width="360px" /> | ||||
|  | ||||
| # Simple, reliable & efficient distributed task queue in Go | ||||
|  | ||||
| [](https://travis-ci.com/hibiken/asynq) | ||||
| [](https://godoc.org/github.com/hibiken/asynq) | ||||
| [](https://goreportcard.com/report/github.com/hibiken/asynq) | ||||
|  | ||||
| [](https://opensource.org/licenses/MIT) | ||||
| [](https://gitter.im/go-asynq/community) | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| Asynq is a Go library for queueing tasks and processing them asynchronously with workers. It's backed by Redis and is designed to be scalable yet easy to get started. | ||||
| Asynq is a Go library for queueing tasks and processing them asynchronously with workers. It's backed by [Redis](https://redis.io/) and is designed to be scalable yet easy to get started. | ||||
|  | ||||
| Highlevel overview of how Asynq works: | ||||
|  | ||||
| - Client puts task on a queue | ||||
| - Server pulls task off queues and starts a worker goroutine for each task | ||||
| - Client puts tasks on a queue | ||||
| - Server pulls tasks off queues and starts a worker goroutine for each task | ||||
| - Tasks are processed concurrently by multiple workers | ||||
|  | ||||
| Task queues are used as a mechanism to distribute work across multiple machines.   | ||||
| A system can consist of multiple worker servers and brokers, giving way to high availability and horizontal scaling. | ||||
| Task queues are used as a mechanism to distribute work across multiple machines. A system can consist of multiple worker servers and brokers, giving way to high availability and horizontal scaling. | ||||
|  | ||||
|  | ||||
| **Example use case** | ||||
|  | ||||
| ## Stability and Compatibility | ||||
|  | ||||
| **Important Note**: Current major version is zero (v0.x.x) to accomodate 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. | ||||
|  | ||||
| **Status**: The library is currently undergoing heavy development with frequent, breaking API changes. | ||||
|  | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| - Guaranteed [at least one execution](https://www.cloudcomputingpatterns.org/at_least_once_delivery/) of a task | ||||
| - Scheduling of tasks | ||||
| - Durability since tasks are written to Redis | ||||
| - [Retries](https://github.com/hibiken/asynq/wiki/Task-Retry) of failed tasks | ||||
| - Automatic recovery of tasks in the event of a worker crash | ||||
| - [Weighted priority queues](https://github.com/hibiken/asynq/wiki/Priority-Queues#weighted-priority-queues) | ||||
| - [Strict priority queues](https://github.com/hibiken/asynq/wiki/Priority-Queues#strict-priority-queues) | ||||
| - [Weighted priority queues](https://github.com/hibiken/asynq/wiki/Queue-Priority#weighted-priority) | ||||
| - [Strict priority queues](https://github.com/hibiken/asynq/wiki/Queue-Priority#strict-priority) | ||||
| - Low latency to add a task since writes are fast in Redis | ||||
| - De-duplication of tasks using [unique option](https://github.com/hibiken/asynq/wiki/Unique-Tasks) | ||||
| - Allow [timeout and deadline per task](https://github.com/hibiken/asynq/wiki/Task-Timeout-and-Cancelation) | ||||
| @@ -44,16 +38,27 @@ A system can consist of multiple worker servers and brokers, giving way to high | ||||
| - [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 | ||||
| - [Web UI](#web-ui) to inspect and remote-control queues and tasks | ||||
| - [CLI](#command-line-tool) to inspect and remote-control queues and tasks | ||||
|  | ||||
| ## Stability and Compatibility | ||||
|  | ||||
| **Status**: The library is currently undergoing **heavy development** with frequent, breaking API changes. | ||||
|  | ||||
| > ☝️ **Important Note**: Current major version is zero (`v0.x.x`) to accomodate 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 | ||||
|  | ||||
| First, make sure you are running a Redis server locally. | ||||
| Make sure you have Go installed ([download](https://golang.org/dl/)). Version `1.14` or higher is required. | ||||
|  | ||||
| Initialize your project by creating a folder and then running `go mod init github.com/your/repo` ([learn more](https://blog.golang.org/using-go-modules)) inside the folder. Then install Asynq library with the [`go get`](https://golang.org/cmd/go/#hdr-Add_dependencies_to_current_module_and_install_them) command: | ||||
|  | ||||
| ```sh | ||||
| $ redis-server | ||||
| go get -u github.com/hibiken/asynq | ||||
| ``` | ||||
|  | ||||
| Make sure you're running a Redis server locally or from a [Docker](https://hub.docker.com/_/redis) container. Version `4.0` or higher is required. | ||||
|  | ||||
| Next, write a package that encapsulates task creation and task handling. | ||||
|  | ||||
| ```go | ||||
| @@ -71,19 +76,35 @@ const ( | ||||
|     TypeImageResize     = "image:resize" | ||||
| ) | ||||
|  | ||||
| type EmailDeliveryPayload struct { | ||||
|     UserID     int | ||||
|     TemplateID string | ||||
| } | ||||
|  | ||||
| type ImageResizePayload struct { | ||||
|     SourceURL string | ||||
| } | ||||
|  | ||||
| //---------------------------------------------- | ||||
| // Write a function NewXXXTask to create a task. | ||||
| // A task consists of a type and a payload. | ||||
| //---------------------------------------------- | ||||
|  | ||||
| func NewEmailDeliveryTask(userID int, tmplID string) *asynq.Task { | ||||
|     payload := map[string]interface{}{"user_id": userID, "template_id": tmplID} | ||||
|     return asynq.NewTask(TypeEmailDelivery, payload) | ||||
| func NewEmailDeliveryTask(userID int, tmplID string) (*asynq.Task, error) { | ||||
|     payload, err := json.Marshal(EmailDeliveryPayload{UserID: userID, TemplateID: tmplID}) | ||||
|     if err != nil { | ||||
|         return nil, err | ||||
|     } | ||||
|     return asynq.NewTask(TypeEmailDelivery, payload), nil | ||||
| } | ||||
|  | ||||
| func NewImageResizeTask(src string) *asynq.Task { | ||||
|     payload := map[string]interface{}{"src": src} | ||||
|     return asynq.NewTask(TypeImageResize, payload) | ||||
| func NewImageResizeTask(src string) (*asynq.Task, error) { | ||||
|     payload, err := json.Marshal(ImageResizePayload{SourceURL: src}) | ||||
|     if err != nil { | ||||
|         return nil, err | ||||
|     } | ||||
|     // task options can be passed to NewTask, which can be overridden at enqueue time. | ||||
|     return asynq.NewTask(TypeImageResize, payload, asynq.MaxRetry(5), asynq.Timeout(20 * time.Minute)), nil | ||||
| } | ||||
|  | ||||
| //--------------------------------------------------------------- | ||||
| @@ -95,15 +116,11 @@ func NewImageResizeTask(src string) *asynq.Task { | ||||
| //--------------------------------------------------------------- | ||||
|  | ||||
| func HandleEmailDeliveryTask(ctx context.Context, t *asynq.Task) error { | ||||
|     userID, err := t.Payload.GetInt("user_id") | ||||
|     if err != nil { | ||||
|         return err | ||||
|     var p EmailDeliveryPayload | ||||
|     if err := json.Unmarshal(t.Payload(), &p); err != nil { | ||||
|         return fmt.Errorf("json.Unmarshal failed: %v: %w", err, asynq.SkipRetry) | ||||
|     } | ||||
|     tmplID, err := t.Payload.GetString("template_id") | ||||
|     if err != nil { | ||||
|         return err | ||||
|     } | ||||
|     fmt.Printf("Send Email to User: user_id = %d, template_id = %s\n", userID, tmplID) | ||||
|     log.Printf("Sending Email to User: user_id=%d, template_id=%s", p.UserID, p.TemplateID) | ||||
|     // Email delivery code ... | ||||
|     return nil | ||||
| } | ||||
| @@ -113,28 +130,27 @@ type ImageProcessor struct { | ||||
|     // ... fields for struct | ||||
| } | ||||
|  | ||||
| func (p *ImageProcessor) ProcessTask(ctx context.Context, t *asynq.Task) error { | ||||
|     src, err := t.Payload.GetString("src") | ||||
|     if err != nil { | ||||
|         return err | ||||
| func (processor *ImageProcessor) ProcessTask(ctx context.Context, t *asynq.Task) error { | ||||
|     var p ImageResizePayload | ||||
|     if err := json.Unmarshal(t.Payload(), &p); err != nil { | ||||
|         return fmt.Errorf("json.Unmarshal failed: %v: %w", err, asynq.SkipRetry) | ||||
|     } | ||||
|     fmt.Printf("Resize image: src = %s\n", src) | ||||
|     log.Printf("Resizing image: src=%s", p.SourceURL) | ||||
|     // Image resizing code ... | ||||
|     return nil | ||||
| } | ||||
|  | ||||
| func NewImageProcessor() *ImageProcessor { | ||||
|     // ... return an instance | ||||
| 	return &ImageProcessor{} | ||||
| } | ||||
| ``` | ||||
|  | ||||
| In your application code, import the above package and use [`Client`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Client) to put tasks on the queue. | ||||
| In your application code, import the above package and use [`Client`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Client) to put tasks on queues. | ||||
|  | ||||
| ```go | ||||
| package main | ||||
|  | ||||
| import ( | ||||
|     "fmt" | ||||
|     "log" | ||||
|     "time" | ||||
|  | ||||
| @@ -145,21 +161,23 @@ import ( | ||||
| const redisAddr = "127.0.0.1:6379" | ||||
|  | ||||
| func main() { | ||||
|     r := asynq.RedisClientOpt{Addr: redisAddr} | ||||
|     c := asynq.NewClient(r) | ||||
|     defer c.Close() | ||||
|     client := asynq.NewClient(asynq.RedisClientOpt{Addr: redisAddr}) | ||||
|     defer client.Close() | ||||
|  | ||||
|     // ------------------------------------------------------ | ||||
|     // Example 1: Enqueue task to be processed immediately. | ||||
|     //            Use (*Client).Enqueue method. | ||||
|     // ------------------------------------------------------ | ||||
|  | ||||
|     t := tasks.NewEmailDeliveryTask(42, "some:template:id") | ||||
|     res, err := c.Enqueue(t) | ||||
|     task, err := tasks.NewEmailDeliveryTask(42, "some:template:id") | ||||
|     if err != nil { | ||||
|         log.Fatal("could not enqueue task: %v", err) | ||||
|         log.Fatalf("could not create task: %v", err) | ||||
|     } | ||||
|     fmt.Printf("Enqueued Result: %+v\n", res) | ||||
|     info, err := client.Enqueue(task) | ||||
|     if err != nil { | ||||
|         log.Fatalf("could not enqueue task: %v", err) | ||||
|     } | ||||
|     log.Printf("enqueued task: id=%s queue=%s", info.ID, info.Queue) | ||||
|  | ||||
|  | ||||
|     // ------------------------------------------------------------ | ||||
| @@ -167,12 +185,11 @@ func main() { | ||||
|     //            Use ProcessIn or ProcessAt option. | ||||
|     // ------------------------------------------------------------ | ||||
|  | ||||
|     t = tasks.NewEmailDeliveryTask(42, "other:template:id") | ||||
|     res, err = c.Enqueue(t, asynq.ProcessIn(24*time.Hour)) | ||||
|     info, err = client.Enqueue(task, asynq.ProcessIn(24*time.Hour)) | ||||
|     if err != nil { | ||||
|         log.Fatal("could not schedule task: %v", err) | ||||
|         log.Fatalf("could not schedule task: %v", err) | ||||
|     } | ||||
|     fmt.Printf("Enqueued Result: %+v\n", res) | ||||
|     log.Printf("enqueued task: id=%s queue=%s", info.ID, info.Queue) | ||||
|  | ||||
|  | ||||
|     // ---------------------------------------------------------------------------- | ||||
| @@ -180,33 +197,21 @@ func main() { | ||||
|     //            Options include MaxRetry, Queue, Timeout, Deadline, Unique etc. | ||||
|     // ---------------------------------------------------------------------------- | ||||
|  | ||||
|     c.SetDefaultOptions(tasks.TypeImageResize, asynq.MaxRetry(10), asynq.Timeout(3*time.Minute)) | ||||
|  | ||||
|     t = tasks.NewImageResizeTask("some/blobstore/path") | ||||
|     res, err = c.Enqueue(t) | ||||
|     task, err = tasks.NewImageResizeTask("https://example.com/myassets/image.jpg") | ||||
|     if err != nil { | ||||
|         log.Fatal("could not enqueue task: %v", err) | ||||
|         log.Fatalf("could not create task: %v", err) | ||||
|     } | ||||
|     fmt.Printf("Enqueued Result: %+v\n", res) | ||||
|  | ||||
|     // --------------------------------------------------------------------------- | ||||
|     // Example 4: Pass options to tune task processing behavior at enqueue time. | ||||
|     //            Options passed at enqueue time override default ones, if any. | ||||
|     // --------------------------------------------------------------------------- | ||||
|  | ||||
|     t = tasks.NewImageResizeTask("some/blobstore/path") | ||||
|     res, err = c.Enqueue(t, asynq.Queue("critical"), asynq.Timeout(30*time.Second)) | ||||
|     info, err = client.Enqueue(task, asynq.MaxRetry(10), asynq.Timeout(3 * time.Minute)) | ||||
|     if err != nil { | ||||
|         log.Fatal("could not enqueue task: %v", err) | ||||
|         log.Fatalf("could not enqueue task: %v", err) | ||||
|     } | ||||
|     fmt.Printf("Enqueued Result: %+v\n", res) | ||||
|     log.Printf("enqueued task: id=%s queue=%s", info.ID, info.Queue) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Next, start a worker server to process these tasks in the background.   | ||||
| To start the background workers, use [`Server`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Server) and provide your [`Handler`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Handler) to process the tasks. | ||||
| Next, start a worker server to process these tasks in the background. To start the background workers, use [`Server`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Server) and provide your [`Handler`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#Handler) to process the tasks. | ||||
|  | ||||
| You can optionally use [`ServeMux`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#ServeMux) to create a handler, just as you would with [`"net/http"`](https://golang.org/pkg/net/http/) Handler. | ||||
| You can optionally use [`ServeMux`](https://pkg.go.dev/github.com/hibiken/asynq?tab=doc#ServeMux) to create a handler, just as you would with [`net/http`](https://golang.org/pkg/net/http/) Handler. | ||||
|  | ||||
| ```go | ||||
| package main | ||||
| @@ -221,9 +226,9 @@ import ( | ||||
| const redisAddr = "127.0.0.1:6379" | ||||
|  | ||||
| func main() { | ||||
|     r := asynq.RedisClientOpt{Addr: redisAddr} | ||||
|  | ||||
|     srv := asynq.NewServer(r, asynq.Config{ | ||||
|     srv := asynq.NewServer( | ||||
|         asynq.RedisClientOpt{Addr: redisAddr}, | ||||
|         asynq.Config{ | ||||
|             // Specify how many concurrent workers to use | ||||
|             Concurrency: 10, | ||||
|             // Optionally specify multiple queues with different priority. | ||||
| @@ -233,7 +238,8 @@ func main() { | ||||
|                 "low":      1, | ||||
|             }, | ||||
|             // See the godoc for other configuration options | ||||
|     }) | ||||
|         }, | ||||
|     ) | ||||
|  | ||||
|     // mux maps a type to a handler | ||||
|     mux := asynq.NewServeMux() | ||||
| @@ -247,52 +253,52 @@ func main() { | ||||
| } | ||||
| ``` | ||||
|  | ||||
| For a more detailed walk-through of the library, see our [Getting Started Guide](https://github.com/hibiken/asynq/wiki/Getting-Started). | ||||
| For a more detailed walk-through of the library, see our [Getting Started](https://github.com/hibiken/asynq/wiki/Getting-Started) guide. | ||||
|  | ||||
| To Learn more about `asynq` features and APIs, see our [Wiki](https://github.com/hibiken/asynq/wiki) and [godoc](https://godoc.org/github.com/hibiken/asynq). | ||||
| To learn more about `asynq` features and APIs, see the package [godoc](https://godoc.org/github.com/hibiken/asynq). | ||||
|  | ||||
| ## Web UI | ||||
|  | ||||
| [Asynqmon](https://github.com/hibiken/asynqmon) is a web based tool for monitoring and administrating Asynq queues and tasks. | ||||
|  | ||||
| Here's a few screenshots of the Web UI: | ||||
|  | ||||
| **Queues view** | ||||
|  | ||||
|  | ||||
|  | ||||
| **Tasks view** | ||||
|  | ||||
|  | ||||
|  | ||||
| **Settings and adaptive dark mode** | ||||
|  | ||||
|  | ||||
|  | ||||
| For details on how to use the tool, refer to the tool's [README](https://github.com/hibiken/asynqmon#readme). | ||||
|  | ||||
| ## Command Line Tool | ||||
|  | ||||
| Asynq ships with a command line tool to inspect the state of queues and tasks. | ||||
|  | ||||
| Here's an example of running the `stats` command. | ||||
|  | ||||
|  | ||||
|  | ||||
| For details on how to use the tool, refer to the tool's [README](/tools/asynq/README.md). | ||||
|  | ||||
| ## Installation | ||||
|  | ||||
| To install `asynq` library, run the following command: | ||||
|  | ||||
| ```sh | ||||
| go get -u github.com/hibiken/asynq | ||||
| ``` | ||||
|  | ||||
| To install the CLI tool, run the following command: | ||||
|  | ||||
| ```sh | ||||
| go get -u github.com/hibiken/asynq/tools/asynq | ||||
| ``` | ||||
|  | ||||
| ## Requirements | ||||
| Here's an example of running the `asynq stats` command: | ||||
|  | ||||
| | Dependency                 | Version | | ||||
| | -------------------------- | ------- | | ||||
| | [Redis](https://redis.io/) | v3.0+   | | ||||
| | [Go](https://golang.org/)  | v1.13+  | | ||||
|  | ||||
|  | ||||
| For details on how to use the tool, refer to the tool's [README](/tools/asynq/README.md). | ||||
|  | ||||
| ## Contributing | ||||
|  | ||||
| We are open to, and grateful for, any contributions (Github issues/pull-requests, feedback on Gitter channel, etc) made by the community. | ||||
| We are open to, and grateful for, any contributions (GitHub issues/PRs, feedback on [Gitter channel](https://gitter.im/go-asynq/community), etc) made by the community. | ||||
|  | ||||
| Please see the [Contribution Guide](/CONTRIBUTING.md) before contributing. | ||||
|  | ||||
| ## Acknowledgements | ||||
|  | ||||
| - [Sidekiq](https://github.com/mperham/sidekiq) : Many of the design ideas are taken from sidekiq and its Web UI | ||||
| - [RQ](https://github.com/rq/rq) : Client APIs are inspired by rq library. | ||||
| - [Cobra](https://github.com/spf13/cobra) : Asynq CLI is built with cobra | ||||
|  | ||||
| ## License | ||||
|  | ||||
| Asynq is released under the MIT license. See [LICENSE](https://github.com/hibiken/asynq/blob/master/LICENSE). | ||||
| Copyright (c) 2019-present [Ken Hibino](https://github.com/hibiken) and [Contributors](https://github.com/hibiken/asynq/graphs/contributors). `Asynq` is free and open-source software licensed under the [MIT License](https://github.com/hibiken/asynq/blob/master/LICENSE). Official logo was created by [Vic Shóstak](https://github.com/koddr) and distributed under [Creative Commons](https://creativecommons.org/publicdomain/zero/1.0/) license (CC0 1.0 Universal). | ||||
|   | ||||
							
								
								
									
										387
									
								
								asynq.go
									
									
									
									
									
								
							
							
						
						
									
										387
									
								
								asynq.go
									
									
									
									
									
								
							| @@ -5,34 +5,201 @@ | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
| 	"fmt" | ||||
| 	"net/url" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| // Task represents a unit of work to be performed. | ||||
| type Task struct { | ||||
| 	// Type indicates the type of task to be performed. | ||||
| 	Type string | ||||
| 	// typename indicates the type of task to be performed. | ||||
| 	typename string | ||||
|  | ||||
| 	// Payload holds data needed to perform the task. | ||||
| 	Payload Payload | ||||
| 	// payload holds data needed to perform the task. | ||||
| 	payload []byte | ||||
|  | ||||
| 	// opts holds options for the task. | ||||
| 	opts []Option | ||||
|  | ||||
| 	// w is the ResultWriter for the task. | ||||
| 	w *ResultWriter | ||||
| } | ||||
|  | ||||
| // NewTask returns a new Task given a type name and payload data. | ||||
| func (t *Task) Type() string    { return t.typename } | ||||
| func (t *Task) Payload() []byte { return t.payload } | ||||
|  | ||||
| // ResultWriter returns a pointer to the ResultWriter associated with the task. | ||||
| // | ||||
| // The payload values must be serializable. | ||||
| func NewTask(typename string, payload map[string]interface{}) *Task { | ||||
| // Nil pointer is returned if called on a newly created task (i.e. task created by calling NewTask). | ||||
| // Only the tasks passed to Handler.ProcessTask have a valid ResultWriter pointer. | ||||
| func (t *Task) ResultWriter() *ResultWriter { return t.w } | ||||
|  | ||||
| // NewTask returns a new Task given a type name and payload data. | ||||
| // Options can be passed to configure task processing behavior. | ||||
| func NewTask(typename string, payload []byte, opts ...Option) *Task { | ||||
| 	return &Task{ | ||||
| 		Type:    typename, | ||||
| 		Payload: Payload{payload}, | ||||
| 		typename: typename, | ||||
| 		payload:  payload, | ||||
| 		opts:     opts, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // newTask creates a task with the given typename, payload and ResultWriter. | ||||
| func newTask(typename string, payload []byte, w *ResultWriter) *Task { | ||||
| 	return &Task{ | ||||
| 		typename: typename, | ||||
| 		payload:  payload, | ||||
| 		w:        w, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // A TaskInfo describes a task and its metadata. | ||||
| type TaskInfo struct { | ||||
| 	// ID is the identifier of the task. | ||||
| 	ID string | ||||
|  | ||||
| 	// Queue is the name of the queue in which the task belongs. | ||||
| 	Queue string | ||||
|  | ||||
| 	// Type is the type name of the task. | ||||
| 	Type string | ||||
|  | ||||
| 	// Payload is the payload data of the task. | ||||
| 	Payload []byte | ||||
|  | ||||
| 	// State indicates the task state. | ||||
| 	State TaskState | ||||
|  | ||||
| 	// MaxRetry is the maximum number of times the task can be retried. | ||||
| 	MaxRetry int | ||||
|  | ||||
| 	// Retried is the number of times the task has retried so far. | ||||
| 	Retried int | ||||
|  | ||||
| 	// LastErr is the error message from the last failure. | ||||
| 	LastErr string | ||||
|  | ||||
| 	// LastFailedAt is the time time of the last failure if any. | ||||
| 	// If the task has no failures, LastFailedAt is zero time (i.e. time.Time{}). | ||||
| 	LastFailedAt time.Time | ||||
|  | ||||
| 	// Timeout is the duration the task can be processed by Handler before being retried, | ||||
| 	// zero if not specified | ||||
| 	Timeout time.Duration | ||||
|  | ||||
| 	// Deadline is the deadline for the task, zero value if not specified. | ||||
| 	Deadline time.Time | ||||
|  | ||||
| 	// NextProcessAt is the time the task is scheduled to be processed, | ||||
| 	// zero if not applicable. | ||||
| 	NextProcessAt time.Time | ||||
|  | ||||
| 	// Retention is duration of the retention period after the task is successfully processed. | ||||
| 	Retention time.Duration | ||||
|  | ||||
| 	// CompletedAt is the time when the task is processed successfully. | ||||
| 	// Zero value (i.e. time.Time{}) indicates no value. | ||||
| 	CompletedAt time.Time | ||||
|  | ||||
| 	// Result holds the result data associated with the task. | ||||
| 	// Use ResultWriter to write result data from the Handler. | ||||
| 	Result []byte | ||||
| } | ||||
|  | ||||
| // If t is non-zero, returns time converted from t as unix time in seconds. | ||||
| // If t is zero, returns zero value of time.Time. | ||||
| func fromUnixTimeOrZero(t int64) time.Time { | ||||
| 	if t == 0 { | ||||
| 		return time.Time{} | ||||
| 	} | ||||
| 	return time.Unix(t, 0) | ||||
| } | ||||
|  | ||||
| func newTaskInfo(msg *base.TaskMessage, state base.TaskState, nextProcessAt time.Time, result []byte) *TaskInfo { | ||||
| 	info := TaskInfo{ | ||||
| 		ID:            msg.ID, | ||||
| 		Queue:         msg.Queue, | ||||
| 		Type:          msg.Type, | ||||
| 		Payload:       msg.Payload, // Do we need to make a copy? | ||||
| 		MaxRetry:      msg.Retry, | ||||
| 		Retried:       msg.Retried, | ||||
| 		LastErr:       msg.ErrorMsg, | ||||
| 		Timeout:       time.Duration(msg.Timeout) * time.Second, | ||||
| 		Deadline:      fromUnixTimeOrZero(msg.Deadline), | ||||
| 		Retention:     time.Duration(msg.Retention) * time.Second, | ||||
| 		NextProcessAt: nextProcessAt, | ||||
| 		LastFailedAt:  fromUnixTimeOrZero(msg.LastFailedAt), | ||||
| 		CompletedAt:   fromUnixTimeOrZero(msg.CompletedAt), | ||||
| 		Result:        result, | ||||
| 	} | ||||
|  | ||||
| 	switch state { | ||||
| 	case base.TaskStateActive: | ||||
| 		info.State = TaskStateActive | ||||
| 	case base.TaskStatePending: | ||||
| 		info.State = TaskStatePending | ||||
| 	case base.TaskStateScheduled: | ||||
| 		info.State = TaskStateScheduled | ||||
| 	case base.TaskStateRetry: | ||||
| 		info.State = TaskStateRetry | ||||
| 	case base.TaskStateArchived: | ||||
| 		info.State = TaskStateArchived | ||||
| 	case base.TaskStateCompleted: | ||||
| 		info.State = TaskStateCompleted | ||||
| 	default: | ||||
| 		panic(fmt.Sprintf("internal error: unknown state: %d", state)) | ||||
| 	} | ||||
| 	return &info | ||||
| } | ||||
|  | ||||
| // TaskState denotes the state of a task. | ||||
| type TaskState int | ||||
|  | ||||
| const ( | ||||
| 	// Indicates that the task is currently being processed by Handler. | ||||
| 	TaskStateActive TaskState = iota + 1 | ||||
|  | ||||
| 	// Indicates that the task is ready to be processed by Handler. | ||||
| 	TaskStatePending | ||||
|  | ||||
| 	// Indicates that the task is scheduled to be processed some time in the future. | ||||
| 	TaskStateScheduled | ||||
|  | ||||
| 	// Indicates that the task has previously failed and scheduled to be processed some time in the future. | ||||
| 	TaskStateRetry | ||||
|  | ||||
| 	// Indicates that the task is archived and stored for inspection purposes. | ||||
| 	TaskStateArchived | ||||
|  | ||||
| 	// Indicates that the task is processed successfully and retained until the retention TTL expires. | ||||
| 	TaskStateCompleted | ||||
| ) | ||||
|  | ||||
| func (s TaskState) String() string { | ||||
| 	switch s { | ||||
| 	case TaskStateActive: | ||||
| 		return "active" | ||||
| 	case TaskStatePending: | ||||
| 		return "pending" | ||||
| 	case TaskStateScheduled: | ||||
| 		return "scheduled" | ||||
| 	case TaskStateRetry: | ||||
| 		return "retry" | ||||
| 	case TaskStateArchived: | ||||
| 		return "archived" | ||||
| 	case TaskStateCompleted: | ||||
| 		return "completed" | ||||
| 	} | ||||
| 	panic("asynq: unknown task state") | ||||
| } | ||||
|  | ||||
| // RedisConnOpt is a discriminated union of types that represent Redis connection configuration option. | ||||
| // | ||||
| // RedisConnOpt represents a sum of following types: | ||||
| @@ -40,7 +207,11 @@ func NewTask(typename string, payload map[string]interface{}) *Task { | ||||
| //   - RedisClientOpt | ||||
| //   - RedisFailoverClientOpt | ||||
| //   - RedisClusterClientOpt | ||||
| type RedisConnOpt interface{} | ||||
| type RedisConnOpt interface { | ||||
| 	// MakeRedisClient returns a new redis client instance. | ||||
| 	// Return value is intentionally opaque to hide the implementation detail of redis client. | ||||
| 	MakeRedisClient() interface{} | ||||
| } | ||||
|  | ||||
| // RedisClientOpt is used to create a redis client that connects | ||||
| // to a redis server directly. | ||||
| @@ -64,6 +235,26 @@ type RedisClientOpt struct { | ||||
| 	// See: https://redis.io/commands/select. | ||||
| 	DB int | ||||
|  | ||||
| 	// Dial timeout for establishing new connections. | ||||
| 	// Default is 5 seconds. | ||||
| 	DialTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket reads. | ||||
| 	// If timeout is reached, read commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is 3 seconds. | ||||
| 	ReadTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket writes. | ||||
| 	// If timeout is reached, write commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is ReadTimout. | ||||
| 	WriteTimeout time.Duration | ||||
|  | ||||
| 	// Maximum number of socket connections. | ||||
| 	// Default is 10 connections per every CPU as reported by runtime.NumCPU. | ||||
| 	PoolSize int | ||||
| @@ -73,6 +264,21 @@ type RedisClientOpt struct { | ||||
| 	TLSConfig *tls.Config | ||||
| } | ||||
|  | ||||
| func (opt RedisClientOpt) MakeRedisClient() interface{} { | ||||
| 	return redis.NewClient(&redis.Options{ | ||||
| 		Network:      opt.Network, | ||||
| 		Addr:         opt.Addr, | ||||
| 		Username:     opt.Username, | ||||
| 		Password:     opt.Password, | ||||
| 		DB:           opt.DB, | ||||
| 		DialTimeout:  opt.DialTimeout, | ||||
| 		ReadTimeout:  opt.ReadTimeout, | ||||
| 		WriteTimeout: opt.WriteTimeout, | ||||
| 		PoolSize:     opt.PoolSize, | ||||
| 		TLSConfig:    opt.TLSConfig, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RedisFailoverClientOpt is used to creates a redis client that talks | ||||
| // to redis sentinels for service discovery and has an automatic failover | ||||
| // capability. | ||||
| @@ -100,6 +306,26 @@ type RedisFailoverClientOpt struct { | ||||
| 	// See: https://redis.io/commands/select. | ||||
| 	DB int | ||||
|  | ||||
| 	// Dial timeout for establishing new connections. | ||||
| 	// Default is 5 seconds. | ||||
| 	DialTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket reads. | ||||
| 	// If timeout is reached, read commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is 3 seconds. | ||||
| 	ReadTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket writes. | ||||
| 	// If timeout is reached, write commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is ReadTimeout | ||||
| 	WriteTimeout time.Duration | ||||
|  | ||||
| 	// Maximum number of socket connections. | ||||
| 	// Default is 10 connections per every CPU as reported by runtime.NumCPU. | ||||
| 	PoolSize int | ||||
| @@ -109,7 +335,23 @@ type RedisFailoverClientOpt struct { | ||||
| 	TLSConfig *tls.Config | ||||
| } | ||||
|  | ||||
| // RedisFailoverClientOpt is used to creates a redis client that connects to | ||||
| func (opt RedisFailoverClientOpt) MakeRedisClient() interface{} { | ||||
| 	return redis.NewFailoverClient(&redis.FailoverOptions{ | ||||
| 		MasterName:       opt.MasterName, | ||||
| 		SentinelAddrs:    opt.SentinelAddrs, | ||||
| 		SentinelPassword: opt.SentinelPassword, | ||||
| 		Username:         opt.Username, | ||||
| 		Password:         opt.Password, | ||||
| 		DB:               opt.DB, | ||||
| 		DialTimeout:      opt.DialTimeout, | ||||
| 		ReadTimeout:      opt.ReadTimeout, | ||||
| 		WriteTimeout:     opt.WriteTimeout, | ||||
| 		PoolSize:         opt.PoolSize, | ||||
| 		TLSConfig:        opt.TLSConfig, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // RedisClusterClientOpt is used to creates a redis client that connects to | ||||
| // redis cluster. | ||||
| type RedisClusterClientOpt struct { | ||||
| 	// A seed list of host:port addresses of cluster nodes. | ||||
| @@ -128,11 +370,44 @@ type RedisClusterClientOpt struct { | ||||
| 	// See: https://redis.io/commands/auth. | ||||
| 	Password string | ||||
|  | ||||
| 	// Dial timeout for establishing new connections. | ||||
| 	// Default is 5 seconds. | ||||
| 	DialTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket reads. | ||||
| 	// If timeout is reached, read commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is 3 seconds. | ||||
| 	ReadTimeout time.Duration | ||||
|  | ||||
| 	// Timeout for socket writes. | ||||
| 	// If timeout is reached, write commands will fail with a timeout error | ||||
| 	// instead of blocking. | ||||
| 	// | ||||
| 	// Use value -1 for no timeout and 0 for default. | ||||
| 	// Default is ReadTimeout. | ||||
| 	WriteTimeout time.Duration | ||||
|  | ||||
| 	// TLS Config used to connect to a server. | ||||
| 	// TLS will be negotiated only if this field is set. | ||||
| 	TLSConfig *tls.Config | ||||
| } | ||||
|  | ||||
| func (opt RedisClusterClientOpt) MakeRedisClient() interface{} { | ||||
| 	return redis.NewClusterClient(&redis.ClusterOptions{ | ||||
| 		Addrs:        opt.Addrs, | ||||
| 		MaxRedirects: opt.MaxRedirects, | ||||
| 		Username:     opt.Username, | ||||
| 		Password:     opt.Password, | ||||
| 		DialTimeout:  opt.DialTimeout, | ||||
| 		ReadTimeout:  opt.ReadTimeout, | ||||
| 		WriteTimeout: opt.WriteTimeout, | ||||
| 		TLSConfig:    opt.TLSConfig, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // ParseRedisURI parses redis uri string and returns RedisConnOpt if uri is valid. | ||||
| // It returns a non-nil error if uri cannot be parsed. | ||||
| // | ||||
| @@ -206,70 +481,26 @@ func parseRedisSentinelURI(u *url.URL) (RedisConnOpt, error) { | ||||
| 	return RedisFailoverClientOpt{MasterName: master, SentinelAddrs: addrs, Password: password}, nil | ||||
| } | ||||
|  | ||||
| // createRedisClient returns a redis client given a redis connection configuration. | ||||
| // | ||||
| // Passing an unexpected type as a RedisConnOpt argument will cause panic. | ||||
| func createRedisClient(r RedisConnOpt) redis.UniversalClient { | ||||
| 	switch r := r.(type) { | ||||
| 	case *RedisClientOpt: | ||||
| 		return redis.NewClient(&redis.Options{ | ||||
| 			Network:   r.Network, | ||||
| 			Addr:      r.Addr, | ||||
| 			Username:  r.Username, | ||||
| 			Password:  r.Password, | ||||
| 			DB:        r.DB, | ||||
| 			PoolSize:  r.PoolSize, | ||||
| 			TLSConfig: r.TLSConfig, | ||||
| 		}) | ||||
| 	case RedisClientOpt: | ||||
| 		return redis.NewClient(&redis.Options{ | ||||
| 			Network:   r.Network, | ||||
| 			Addr:      r.Addr, | ||||
| 			Username:  r.Username, | ||||
| 			Password:  r.Password, | ||||
| 			DB:        r.DB, | ||||
| 			PoolSize:  r.PoolSize, | ||||
| 			TLSConfig: r.TLSConfig, | ||||
| 		}) | ||||
| 	case *RedisFailoverClientOpt: | ||||
| 		return redis.NewFailoverClient(&redis.FailoverOptions{ | ||||
| 			MasterName:       r.MasterName, | ||||
| 			SentinelAddrs:    r.SentinelAddrs, | ||||
| 			SentinelPassword: r.SentinelPassword, | ||||
| 			Username:         r.Username, | ||||
| 			Password:         r.Password, | ||||
| 			DB:               r.DB, | ||||
| 			PoolSize:         r.PoolSize, | ||||
| 			TLSConfig:        r.TLSConfig, | ||||
| 		}) | ||||
| 	case RedisFailoverClientOpt: | ||||
| 		return redis.NewFailoverClient(&redis.FailoverOptions{ | ||||
| 			MasterName:       r.MasterName, | ||||
| 			SentinelAddrs:    r.SentinelAddrs, | ||||
| 			SentinelPassword: r.SentinelPassword, | ||||
| 			Username:         r.Username, | ||||
| 			Password:         r.Password, | ||||
| 			DB:               r.DB, | ||||
| 			PoolSize:         r.PoolSize, | ||||
| 			TLSConfig:        r.TLSConfig, | ||||
| 		}) | ||||
| 	case RedisClusterClientOpt: | ||||
| 		return redis.NewClusterClient(&redis.ClusterOptions{ | ||||
| 			Addrs:        r.Addrs, | ||||
| 			MaxRedirects: r.MaxRedirects, | ||||
| 			Username:     r.Username, | ||||
| 			Password:     r.Password, | ||||
| 			TLSConfig:    r.TLSConfig, | ||||
| 		}) | ||||
| 	case *RedisClusterClientOpt: | ||||
| 		return redis.NewClusterClient(&redis.ClusterOptions{ | ||||
| 			Addrs:        r.Addrs, | ||||
| 			MaxRedirects: r.MaxRedirects, | ||||
| 			Username:     r.Username, | ||||
| 			Password:     r.Password, | ||||
| 			TLSConfig:    r.TLSConfig, | ||||
| 		}) | ||||
| 	default: | ||||
| 		panic(fmt.Sprintf("asynq: unexpected type %T for RedisConnOpt", r)) | ||||
| 	} | ||||
| // ResultWriter is a client interface to write result data for a task. | ||||
| // It writes the data to the redis instance the server is connected to. | ||||
| type ResultWriter struct { | ||||
| 	id     string // task ID this writer is responsible for | ||||
| 	qname  string // queue name the task belongs to | ||||
| 	broker base.Broker | ||||
| 	ctx    context.Context // context associated with the task | ||||
| } | ||||
|  | ||||
| // Write writes the given data as a result of the task the ResultWriter is associated with. | ||||
| func (w *ResultWriter) Write(data []byte) (n int, err error) { | ||||
| 	select { | ||||
| 	case <-w.ctx.Done(): | ||||
| 		return 0, fmt.Errorf("failed to result task result: %v", w.ctx.Err()) | ||||
| 	default: | ||||
| 	} | ||||
| 	return w.broker.WriteResult(w.qname, w.id, data) | ||||
| } | ||||
|  | ||||
| // TaskID returns the ID of the task the ResultWriter is associated with. | ||||
| func (w *ResultWriter) TaskID() string { | ||||
| 	return w.id | ||||
| } | ||||
|   | ||||
| @@ -10,7 +10,7 @@ import ( | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	h "github.com/hibiken/asynq/internal/asynqtest" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| @@ -85,7 +85,7 @@ func getRedisConnOpt(tb testing.TB) RedisConnOpt { | ||||
| var sortTaskOpt = cmp.Transformer("SortMsg", func(in []*Task) []*Task { | ||||
| 	out := append([]*Task(nil), in...) // Copy input to avoid mutating it | ||||
| 	sort.Slice(out, func(i, j int) bool { | ||||
| 		return out[i].Type < out[j].Type | ||||
| 		return out[i].Type() < out[j].Type() | ||||
| 	}) | ||||
| 	return out | ||||
| }) | ||||
|   | ||||
| @@ -6,12 +6,24 @@ package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	h "github.com/hibiken/asynq/internal/asynqtest" | ||||
| ) | ||||
|  | ||||
| // Creates a new task of type "task<n>" with payload {"data": n}. | ||||
| func makeTask(n int) *Task { | ||||
| 	b, err := json.Marshal(map[string]int{"data": n}) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	return NewTask(fmt.Sprintf("task%d", n), b) | ||||
| } | ||||
|  | ||||
| // Simple E2E Benchmark testing with no scheduled tasks and retries. | ||||
| func BenchmarkEndToEndSimple(b *testing.B) { | ||||
| 	const count = 100000 | ||||
| @@ -29,8 +41,7 @@ func BenchmarkEndToEndSimple(b *testing.B) { | ||||
| 		}) | ||||
| 		// Create a bunch of tasks | ||||
| 		for i := 0; i < count; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| @@ -70,14 +81,12 @@ func BenchmarkEndToEnd(b *testing.B) { | ||||
| 		}) | ||||
| 		// Create a bunch of tasks | ||||
| 		for i := 0; i < count; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 		for i := 0; i < count; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("scheduled%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t, ProcessIn(1*time.Second)); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i), ProcessIn(1*time.Second)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| @@ -86,13 +95,18 @@ func BenchmarkEndToEnd(b *testing.B) { | ||||
| 		var wg sync.WaitGroup | ||||
| 		wg.Add(count * 2) | ||||
| 		handler := func(ctx context.Context, t *Task) error { | ||||
| 			n, err := t.Payload.GetInt("data") | ||||
| 			if err != nil { | ||||
| 			var p map[string]int | ||||
| 			if err := json.Unmarshal(t.Payload(), &p); err != nil { | ||||
| 				b.Logf("internal error: %v", err) | ||||
| 			} | ||||
| 			n, ok := p["data"] | ||||
| 			if !ok { | ||||
| 				n = 1 | ||||
| 				b.Logf("internal error: could not get data from payload") | ||||
| 			} | ||||
| 			retried, ok := GetRetryCount(ctx) | ||||
| 			if !ok { | ||||
| 				b.Logf("internal error: %v", err) | ||||
| 				b.Logf("internal error: could not get retry count from context") | ||||
| 			} | ||||
| 			// Fail 1% of tasks for the first attempt. | ||||
| 			if retried == 0 && n%100 == 0 { | ||||
| @@ -136,20 +150,17 @@ func BenchmarkEndToEndMultipleQueues(b *testing.B) { | ||||
| 		}) | ||||
| 		// Create a bunch of tasks | ||||
| 		for i := 0; i < highCount; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t, Queue("high")); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i), Queue("high")); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 		for i := 0; i < defaultCount; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 		for i := 0; i < lowCount; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t, Queue("low")); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i), Queue("low")); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| @@ -190,15 +201,13 @@ func BenchmarkClientWhileServerRunning(b *testing.B) { | ||||
| 		}) | ||||
| 		// Enqueue 10,000 tasks. | ||||
| 		for i := 0; i < count; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("task%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 		// Schedule 10,000 tasks. | ||||
| 		for i := 0; i < count; i++ { | ||||
| 			t := NewTask(fmt.Sprintf("scheduled%d", i), map[string]interface{}{"data": i}) | ||||
| 			if _, err := client.Enqueue(t, ProcessIn(1*time.Second)); err != nil { | ||||
| 			if _, err := client.Enqueue(makeTask(i), ProcessIn(1*time.Second)); err != nil { | ||||
| 				b.Fatalf("could not enqueue a task: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| @@ -213,7 +222,7 @@ func BenchmarkClientWhileServerRunning(b *testing.B) { | ||||
| 		b.Log("Starting enqueueing") | ||||
| 		enqueued := 0 | ||||
| 		for enqueued < 100000 { | ||||
| 			t := NewTask(fmt.Sprintf("enqueued%d", enqueued), map[string]interface{}{"data": enqueued}) | ||||
| 			t := NewTask(fmt.Sprintf("enqueued%d", enqueued), h.JSON(map[string]interface{}{"data": enqueued})) | ||||
| 			if _, err := client.Enqueue(t); err != nil { | ||||
| 				b.Logf("could not enqueue task %d: %v", enqueued, err) | ||||
| 				continue | ||||
|   | ||||
							
								
								
									
										222
									
								
								client.go
									
									
									
									
									
								
							
							
						
						
									
										222
									
								
								client.go
									
									
									
									
									
								
							| @@ -5,15 +5,14 @@ | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| ) | ||||
|  | ||||
| @@ -24,18 +23,16 @@ import ( | ||||
| // | ||||
| // Clients are safe for concurrent use by multiple goroutines. | ||||
| type Client struct { | ||||
| 	mu   sync.Mutex | ||||
| 	opts map[string][]Option | ||||
| 	rdb *rdb.RDB | ||||
| } | ||||
|  | ||||
| // NewClient returns a new Client instance given a redis connection option. | ||||
| func NewClient(r RedisConnOpt) *Client { | ||||
| 	rdb := rdb.NewRDB(createRedisClient(r)) | ||||
| 	return &Client{ | ||||
| 		opts: make(map[string][]Option), | ||||
| 		rdb:  rdb, | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	return &Client{rdb: rdb.NewRDB(c)} | ||||
| } | ||||
|  | ||||
| type OptionType int | ||||
| @@ -48,6 +45,8 @@ const ( | ||||
| 	UniqueOpt | ||||
| 	ProcessAtOpt | ||||
| 	ProcessInOpt | ||||
| 	TaskIDOpt | ||||
| 	RetentionOpt | ||||
| ) | ||||
|  | ||||
| // Option specifies the task processing behavior. | ||||
| @@ -66,11 +65,13 @@ type Option interface { | ||||
| type ( | ||||
| 	retryOption     int | ||||
| 	queueOption     string | ||||
| 	taskIDOption    string | ||||
| 	timeoutOption   time.Duration | ||||
| 	deadlineOption  time.Time | ||||
| 	uniqueOption    time.Duration | ||||
| 	processAtOption time.Time | ||||
| 	processInOption time.Duration | ||||
| 	retentionOption time.Duration | ||||
| ) | ||||
|  | ||||
| // MaxRetry returns an option to specify the max number of times | ||||
| @@ -89,16 +90,23 @@ func (n retryOption) Type() OptionType   { return MaxRetryOpt } | ||||
| func (n retryOption) Value() interface{} { return int(n) } | ||||
|  | ||||
| // Queue returns an option to specify the queue to enqueue the task into. | ||||
| // | ||||
| // Queue name is case-insensitive and the lowercased version is used. | ||||
| func Queue(qname string) Option { | ||||
| 	return queueOption(strings.ToLower(qname)) | ||||
| 	return queueOption(qname) | ||||
| } | ||||
|  | ||||
| func (qname queueOption) String() string     { return fmt.Sprintf("Queue(%q)", string(qname)) } | ||||
| func (qname queueOption) Type() OptionType   { return QueueOpt } | ||||
| func (qname queueOption) Value() interface{} { return string(qname) } | ||||
|  | ||||
| // TaskID returns an option to specify the task ID. | ||||
| func TaskID(id string) Option { | ||||
| 	return taskIDOption(id) | ||||
| } | ||||
|  | ||||
| func (id taskIDOption) String() string     { return fmt.Sprintf("TaskID(%q)", string(id)) } | ||||
| func (id taskIDOption) Type() OptionType   { return TaskIDOpt } | ||||
| func (id taskIDOption) Value() interface{} { return string(id) } | ||||
|  | ||||
| // Timeout returns an option to specify how long a task may run. | ||||
| // If the timeout elapses before the Handler returns, then the task | ||||
| // will be retried. | ||||
| @@ -172,86 +180,36 @@ func (d processInOption) String() string     { return fmt.Sprintf("ProcessIn(%v) | ||||
| func (d processInOption) Type() OptionType   { return ProcessInOpt } | ||||
| func (d processInOption) Value() interface{} { return time.Duration(d) } | ||||
|  | ||||
| // parseOption interprets a string s as an Option and returns the Option if parsing is successful, | ||||
| // otherwise returns non-nil error. | ||||
| func parseOption(s string) (Option, error) { | ||||
| 	fn, arg := parseOptionFunc(s), parseOptionArg(s) | ||||
| 	switch fn { | ||||
| 	case "Queue": | ||||
| 		qname, err := strconv.Unquote(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Queue(qname), nil | ||||
| 	case "MaxRetry": | ||||
| 		n, err := strconv.Atoi(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return MaxRetry(n), nil | ||||
| 	case "Timeout": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Timeout(d), nil | ||||
| 	case "Deadline": | ||||
| 		t, err := time.Parse(time.UnixDate, arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Deadline(t), nil | ||||
| 	case "Unique": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Unique(d), nil | ||||
| 	case "ProcessAt": | ||||
| 		t, err := time.Parse(time.UnixDate, arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return ProcessAt(t), nil | ||||
| 	case "ProcessIn": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return ProcessIn(d), nil | ||||
| 	default: | ||||
| 		return nil, fmt.Errorf("cannot not parse option string %q", s) | ||||
| 	} | ||||
| // Retention returns an option to specify the duration of retention period for the task. | ||||
| // If this option is provided, the task will be stored as a completed task after successful processing. | ||||
| // A completed task will be deleted after the specified duration elapses. | ||||
| func Retention(d time.Duration) Option { | ||||
| 	return retentionOption(d) | ||||
| } | ||||
|  | ||||
| func parseOptionFunc(s string) string { | ||||
| 	i := strings.Index(s, "(") | ||||
| 	return s[:i] | ||||
| } | ||||
|  | ||||
| func parseOptionArg(s string) string { | ||||
| 	i := strings.Index(s, "(") | ||||
| 	if i >= 0 { | ||||
| 		j := strings.Index(s, ")") | ||||
| 		if j > i { | ||||
| 			return s[i+1 : j] | ||||
| 		} | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
| func (ttl retentionOption) String() string     { return fmt.Sprintf("Retention(%v)", time.Duration(ttl)) } | ||||
| func (ttl retentionOption) Type() OptionType   { return RetentionOpt } | ||||
| func (ttl retentionOption) Value() interface{} { return time.Duration(ttl) } | ||||
|  | ||||
| // ErrDuplicateTask indicates that the given task could not be enqueued since it's a duplicate of another task. | ||||
| // | ||||
| // ErrDuplicateTask error only applies to tasks enqueued with a Unique option. | ||||
| var ErrDuplicateTask = errors.New("task already exists") | ||||
|  | ||||
| // ErrTaskIDConflict indicates that the given task could not be enqueued since its task ID already exists. | ||||
| // | ||||
| // ErrTaskIDConflict error only applies to tasks enqueued with a TaskID option. | ||||
| var ErrTaskIDConflict = errors.New("task ID conflicts with another task") | ||||
|  | ||||
| type option struct { | ||||
| 	retry     int | ||||
| 	queue     string | ||||
| 	taskID    string | ||||
| 	timeout   time.Duration | ||||
| 	deadline  time.Time | ||||
| 	uniqueTTL time.Duration | ||||
| 	processAt time.Time | ||||
| 	retention time.Duration | ||||
| } | ||||
|  | ||||
| // composeOptions merges user provided options into the default options | ||||
| @@ -262,6 +220,7 @@ func composeOptions(opts ...Option) (option, error) { | ||||
| 	res := option{ | ||||
| 		retry:     defaultMaxRetry, | ||||
| 		queue:     base.DefaultQueueName, | ||||
| 		taskID:    uuid.NewString(), | ||||
| 		timeout:   0, // do not set to deafultTimeout here | ||||
| 		deadline:  time.Time{}, | ||||
| 		processAt: time.Now(), | ||||
| @@ -271,11 +230,17 @@ func composeOptions(opts ...Option) (option, error) { | ||||
| 		case retryOption: | ||||
| 			res.retry = int(opt) | ||||
| 		case queueOption: | ||||
| 			trimmed := strings.TrimSpace(string(opt)) | ||||
| 			if err := validateQueueName(trimmed); err != nil { | ||||
| 			qname := string(opt) | ||||
| 			if err := base.ValidateQueueName(qname); err != nil { | ||||
| 				return option{}, err | ||||
| 			} | ||||
| 			res.queue = trimmed | ||||
| 			res.queue = qname | ||||
| 		case taskIDOption: | ||||
| 			id := string(opt) | ||||
| 			if err := validateTaskID(id); err != nil { | ||||
| 				return option{}, err | ||||
| 			} | ||||
| 			res.taskID = id | ||||
| 		case timeoutOption: | ||||
| 			res.timeout = time.Duration(opt) | ||||
| 		case deadlineOption: | ||||
| @@ -286,6 +251,8 @@ func composeOptions(opts ...Option) (option, error) { | ||||
| 			res.processAt = time.Time(opt) | ||||
| 		case processInOption: | ||||
| 			res.processAt = time.Now().Add(time.Duration(opt)) | ||||
| 		case retentionOption: | ||||
| 			res.retention = time.Duration(opt) | ||||
| 		default: | ||||
| 			// ignore unexpected option | ||||
| 		} | ||||
| @@ -293,9 +260,10 @@ func composeOptions(opts ...Option) (option, error) { | ||||
| 	return res, nil | ||||
| } | ||||
|  | ||||
| func validateQueueName(qname string) error { | ||||
| 	if len(qname) == 0 { | ||||
| 		return fmt.Errorf("queue name must contain one or more characters") | ||||
| // validates user provided task ID string. | ||||
| func validateTaskID(id string) error { | ||||
| 	if strings.TrimSpace(id) == "" { | ||||
| 		return errors.New("task ID cannot be empty") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @@ -314,52 +282,6 @@ var ( | ||||
| 	noDeadline time.Time     = time.Unix(0, 0) | ||||
| ) | ||||
|  | ||||
| // SetDefaultOptions sets options to be used for a given task type. | ||||
| // The argument opts specifies the behavior of task processing. | ||||
| // If there are conflicting Option values the last one overrides others. | ||||
| // | ||||
| // Default options can be overridden by options passed at enqueue time. | ||||
| func (c *Client) SetDefaultOptions(taskType string, opts ...Option) { | ||||
| 	c.mu.Lock() | ||||
| 	defer c.mu.Unlock() | ||||
| 	c.opts[taskType] = opts | ||||
| } | ||||
|  | ||||
| // A Result holds enqueued task's metadata. | ||||
| type Result struct { | ||||
| 	// ID is a unique identifier for the task. | ||||
| 	ID string | ||||
|  | ||||
| 	// EnqueuedAt is the time the task was enqueued in UTC. | ||||
| 	EnqueuedAt time.Time | ||||
|  | ||||
| 	// ProcessAt indicates when the task should be processed. | ||||
| 	ProcessAt time.Time | ||||
|  | ||||
| 	// Retry is the maximum number of retry for the task. | ||||
| 	Retry int | ||||
|  | ||||
| 	// Queue is a name of the queue the task is enqueued to. | ||||
| 	Queue string | ||||
|  | ||||
| 	// Timeout is the timeout value for the task. | ||||
| 	// Counting for timeout starts when a worker starts processing the task. | ||||
| 	// If task processing doesn't complete within the timeout, the task will be retried. | ||||
| 	// The value zero means no timeout. | ||||
| 	// | ||||
| 	// If deadline is set, min(now+timeout, deadline) is used, where the now is the time when | ||||
| 	// a worker starts processing the task. | ||||
| 	Timeout time.Duration | ||||
|  | ||||
| 	// Deadline is the deadline value for the task. | ||||
| 	// If task processing doesn't complete before the deadline, the task will be retried. | ||||
| 	// The value time.Unix(0, 0) means no deadline. | ||||
| 	// | ||||
| 	// If timeout is set, min(now+timeout, deadline) is used, where the now is the time when | ||||
| 	// a worker starts processing the task. | ||||
| 	Deadline time.Time | ||||
| } | ||||
|  | ||||
| // Close closes the connection with redis. | ||||
| func (c *Client) Close() error { | ||||
| 	return c.rdb.Close() | ||||
| @@ -367,18 +289,20 @@ func (c *Client) Close() error { | ||||
|  | ||||
| // Enqueue enqueues the given task to be processed asynchronously. | ||||
| // | ||||
| // Enqueue returns nil if the task is enqueued successfully, otherwise returns a non-nil error. | ||||
| // Enqueue returns TaskInfo and nil error if the task is enqueued successfully, otherwise returns a non-nil error. | ||||
| // | ||||
| // The argument opts specifies the behavior of task processing. | ||||
| // If there are conflicting Option values the last one overrides others. | ||||
| // Any options provided to NewTask can be overridden by options passed to Enqueue. | ||||
| // By deafult, max retry is set to 25 and timeout is set to 30 minutes. | ||||
| // If no ProcessAt or ProcessIn options are passed, the task will be processed immediately. | ||||
| func (c *Client) Enqueue(task *Task, opts ...Option) (*Result, error) { | ||||
| 	c.mu.Lock() | ||||
| 	if defaults, ok := c.opts[task.Type]; ok { | ||||
| 		opts = append(defaults, opts...) | ||||
| // | ||||
| // If no ProcessAt or ProcessIn options are provided, the task will be pending immediately. | ||||
| func (c *Client) Enqueue(task *Task, opts ...Option) (*TaskInfo, error) { | ||||
| 	if strings.TrimSpace(task.Type()) == "" { | ||||
| 		return nil, fmt.Errorf("task typename cannot be empty") | ||||
| 	} | ||||
| 	c.mu.Unlock() | ||||
| 	// merge task options with the options provided at enqueue time. | ||||
| 	opts = append(task.opts, opts...) | ||||
| 	opt, err := composeOptions(opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| @@ -397,40 +321,38 @@ func (c *Client) Enqueue(task *Task, opts ...Option) (*Result, error) { | ||||
| 	} | ||||
| 	var uniqueKey string | ||||
| 	if opt.uniqueTTL > 0 { | ||||
| 		uniqueKey = base.UniqueKey(opt.queue, task.Type, task.Payload.data) | ||||
| 		uniqueKey = base.UniqueKey(opt.queue, task.Type(), task.Payload()) | ||||
| 	} | ||||
| 	msg := &base.TaskMessage{ | ||||
| 		ID:        uuid.New(), | ||||
| 		Type:      task.Type, | ||||
| 		Payload:   task.Payload.data, | ||||
| 		ID:        opt.taskID, | ||||
| 		Type:      task.Type(), | ||||
| 		Payload:   task.Payload(), | ||||
| 		Queue:     opt.queue, | ||||
| 		Retry:     opt.retry, | ||||
| 		Deadline:  deadline.Unix(), | ||||
| 		Timeout:   int64(timeout.Seconds()), | ||||
| 		UniqueKey: uniqueKey, | ||||
| 		Retention: int64(opt.retention.Seconds()), | ||||
| 	} | ||||
| 	now := time.Now() | ||||
| 	var state base.TaskState | ||||
| 	if opt.processAt.Before(now) || opt.processAt.Equal(now) { | ||||
| 		opt.processAt = now | ||||
| 		err = c.enqueue(msg, opt.uniqueTTL) | ||||
| 		state = base.TaskStatePending | ||||
| 	} else { | ||||
| 		err = c.schedule(msg, opt.processAt, opt.uniqueTTL) | ||||
| 		state = base.TaskStateScheduled | ||||
| 	} | ||||
| 	switch { | ||||
| 	case err == rdb.ErrDuplicateTask: | ||||
| 	case errors.Is(err, errors.ErrDuplicateTask): | ||||
| 		return nil, fmt.Errorf("%w", ErrDuplicateTask) | ||||
| 	case errors.Is(err, errors.ErrTaskIdConflict): | ||||
| 		return nil, fmt.Errorf("%w", ErrTaskIDConflict) | ||||
| 	case err != nil: | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &Result{ | ||||
| 		ID:         msg.ID.String(), | ||||
| 		EnqueuedAt: time.Now().UTC(), | ||||
| 		ProcessAt:  opt.processAt, | ||||
| 		Queue:      msg.Queue, | ||||
| 		Retry:      msg.Retry, | ||||
| 		Timeout:    timeout, | ||||
| 		Deadline:   deadline, | ||||
| 	}, nil | ||||
| 	return newTaskInfo(msg, state, opt.processAt, nil), nil | ||||
| } | ||||
|  | ||||
| func (c *Client) enqueue(msg *base.TaskMessage, uniqueTTL time.Duration) error { | ||||
|   | ||||
							
								
								
									
										562
									
								
								client_test.go
									
									
									
									
									
								
							
							
						
						
									
										562
									
								
								client_test.go
									
									
									
									
									
								
							| @@ -5,6 +5,7 @@ | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @@ -20,7 +21,7 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"}) | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
|  | ||||
| 	var ( | ||||
| 		now          = time.Now() | ||||
| @@ -32,7 +33,7 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 		task          *Task | ||||
| 		processAt     time.Time // value for ProcessAt option | ||||
| 		opts          []Option  // other options | ||||
| 		wantRes       *Result | ||||
| 		wantInfo      *TaskInfo | ||||
| 		wantPending   map[string][]*base.TaskMessage | ||||
| 		wantScheduled map[string][]base.Z | ||||
| 	}{ | ||||
| @@ -41,19 +42,24 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 			task:      task, | ||||
| 			processAt: now, | ||||
| 			opts:      []Option{}, | ||||
| 			wantRes: &Result{ | ||||
| 				EnqueuedAt: now.UTC(), | ||||
| 				ProcessAt:  now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:      defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:   noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -70,13 +76,18 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 			task:      task, | ||||
| 			processAt: oneHourLater, | ||||
| 			opts:      []Option{}, | ||||
| 			wantRes: &Result{ | ||||
| 				EnqueuedAt: now.UTC(), | ||||
| 				ProcessAt:  oneHourLater, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:      defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStateScheduled, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:   noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: oneHourLater, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": {}, | ||||
| @@ -85,8 +96,8 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Message: &base.TaskMessage{ | ||||
| 							Type:     task.Type, | ||||
| 							Payload:  task.Payload.data, | ||||
| 							Type:     task.Type(), | ||||
| 							Payload:  task.Payload(), | ||||
| 							Retry:    defaultMaxRetry, | ||||
| 							Queue:    "default", | ||||
| 							Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -103,24 +114,24 @@ func TestClientEnqueueWithProcessAtOption(t *testing.T) { | ||||
| 		h.FlushDB(t, r) // clean up db before each test case. | ||||
|  | ||||
| 		opts := append(tc.opts, ProcessAt(tc.processAt)) | ||||
| 		gotRes, err := client.Enqueue(tc.task, opts...) | ||||
| 		gotInfo, err := client.Enqueue(tc.task, opts...) | ||||
| 		if err != nil { | ||||
| 			t.Error(err) | ||||
| 			continue | ||||
| 		} | ||||
| 		cmpOptions := []cmp.Option{ | ||||
| 			cmpopts.IgnoreFields(Result{}, "ID"), | ||||
| 			cmpopts.IgnoreFields(TaskInfo{}, "ID"), | ||||
| 			cmpopts.EquateApproxTime(500 * time.Millisecond), | ||||
| 		} | ||||
| 		if diff := cmp.Diff(tc.wantRes, gotRes, cmpOptions...); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantInfo, gotInfo, cmpOptions...); diff != "" { | ||||
| 			t.Errorf("%s;\nEnqueue(task, ProcessAt(%v)) returned %v, want %v; (-want,+got)\n%s", | ||||
| 				tc.desc, tc.processAt, gotRes, tc.wantRes, diff) | ||||
| 				tc.desc, tc.processAt, gotInfo, tc.wantInfo, diff) | ||||
| 		} | ||||
|  | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			gotPending := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotPending, h.IgnoreIDOpt, cmpopts.EquateEmpty()); diff != "" { | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.QueueKey(qname), diff) | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.PendingKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 		for qname, want := range tc.wantScheduled { | ||||
| @@ -137,14 +148,14 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"}) | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc        string | ||||
| 		task        *Task | ||||
| 		opts        []Option | ||||
| 		wantRes     *Result | ||||
| 		wantInfo    *TaskInfo | ||||
| 		wantPending map[string][]*base.TaskMessage | ||||
| 	}{ | ||||
| 		{ | ||||
| @@ -153,18 +164,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			opts: []Option{ | ||||
| 				MaxRetry(3), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     3, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      3, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    3, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -179,18 +196,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			opts: []Option{ | ||||
| 				MaxRetry(-2), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     0, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      0, // Retry count should be set to zero | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    0, // Retry count should be set to zero | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -206,18 +229,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 				MaxRetry(2), | ||||
| 				MaxRetry(10), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     10, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      10, // Last option takes precedence | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    10, // Last option takes precedence | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -232,18 +261,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			opts: []Option{ | ||||
| 				Queue("custom"), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "custom", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"custom": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "custom", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -253,25 +288,31 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "Queue option should be case-insensitive", | ||||
| 			desc: "Queue option should be case sensitive", | ||||
| 			task: task, | ||||
| 			opts: []Option{ | ||||
| 				Queue("HIGH"), | ||||
| 				Queue("MyQueue"), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 				Queue:     "high", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "MyQueue", | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"high": { | ||||
| 				"MyQueue": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "high", | ||||
| 						Queue:    "MyQueue", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| 						Deadline: noDeadline.Unix(), | ||||
| 					}, | ||||
| @@ -284,18 +325,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			opts: []Option{ | ||||
| 				Timeout(20 * time.Second), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       20 * time.Second, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  20, | ||||
| @@ -310,18 +357,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 			opts: []Option{ | ||||
| 				Deadline(time.Date(2020, time.June, 24, 0, 0, 0, 0, time.UTC)), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       noTimeout, | ||||
| 				Deadline:      time.Date(2020, time.June, 24, 0, 0, 0, 0, time.UTC), | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(noTimeout.Seconds()), | ||||
| @@ -337,18 +390,24 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 				Timeout(20 * time.Second), | ||||
| 				Deadline(time.Date(2020, time.June, 24, 0, 0, 0, 0, time.UTC)), | ||||
| 			}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       20 * time.Second, | ||||
| 				Deadline:      time.Date(2020, time.June, 24, 0, 0, 0, 0, time.UTC), | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  20, | ||||
| @@ -357,40 +416,168 @@ func TestClientEnqueue(t *testing.T) { | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With Retention option", | ||||
| 			task: task, | ||||
| 			opts: []Option{ | ||||
| 				Retention(24 * time.Hour), | ||||
| 			}, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 				Retention:     24 * time.Hour, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:      task.Type(), | ||||
| 						Payload:   task.Payload(), | ||||
| 						Retry:     defaultMaxRetry, | ||||
| 						Queue:     "default", | ||||
| 						Timeout:   int64(defaultTimeout.Seconds()), | ||||
| 						Deadline:  noDeadline.Unix(), | ||||
| 						Retention: int64((24 * time.Hour).Seconds()), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r) // clean up db before each test case. | ||||
|  | ||||
| 		gotRes, err := client.Enqueue(tc.task, tc.opts...) | ||||
| 		gotInfo, err := client.Enqueue(tc.task, tc.opts...) | ||||
| 		if err != nil { | ||||
| 			t.Error(err) | ||||
| 			continue | ||||
| 		} | ||||
| 		cmpOptions := []cmp.Option{ | ||||
| 			cmpopts.IgnoreFields(Result{}, "ID", "EnqueuedAt"), | ||||
| 			cmpopts.IgnoreFields(TaskInfo{}, "ID"), | ||||
| 			cmpopts.EquateApproxTime(500 * time.Millisecond), | ||||
| 		} | ||||
| 		if diff := cmp.Diff(tc.wantRes, gotRes, cmpOptions...); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantInfo, gotInfo, cmpOptions...); diff != "" { | ||||
| 			t.Errorf("%s;\nEnqueue(task) returned %v, want %v; (-want,+got)\n%s", | ||||
| 				tc.desc, gotRes, tc.wantRes, diff) | ||||
| 				tc.desc, gotInfo, tc.wantInfo, diff) | ||||
| 		} | ||||
|  | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			got := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, got, h.IgnoreIDOpt); diff != "" { | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.QueueKey(qname), diff) | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.PendingKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueueWithTaskIDOption(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", nil) | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc        string | ||||
| 		task        *Task | ||||
| 		opts        []Option | ||||
| 		wantInfo    *TaskInfo | ||||
| 		wantPending map[string][]*base.TaskMessage | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "With a valid TaskID option", | ||||
| 			task: task, | ||||
| 			opts: []Option{ | ||||
| 				TaskID("custom_id"), | ||||
| 			}, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				ID:            "custom_id", | ||||
| 				Queue:         "default", | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						ID:       "custom_id", | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| 						Deadline: noDeadline.Unix(), | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r) // clean up db before each test case. | ||||
|  | ||||
| 		gotInfo, err := client.Enqueue(tc.task, tc.opts...) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("got non-nil error %v, want nil", err) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		cmpOptions := []cmp.Option{ | ||||
| 			cmpopts.EquateApproxTime(500 * time.Millisecond), | ||||
| 		} | ||||
| 		if diff := cmp.Diff(tc.wantInfo, gotInfo, cmpOptions...); diff != "" { | ||||
| 			t.Errorf("%s;\nEnqueue(task) returned %v, want %v; (-want,+got)\n%s", | ||||
| 				tc.desc, gotInfo, tc.wantInfo, diff) | ||||
| 		} | ||||
|  | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			got := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, got); diff != "" { | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.PendingKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueueWithConflictingTaskID(t *testing.T) { | ||||
| 	setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	const taskID = "custom_id" | ||||
| 	task := NewTask("foo", nil) | ||||
|  | ||||
| 	if _, err := client.Enqueue(task, TaskID(taskID)); err != nil { | ||||
| 		t.Fatalf("First task: Enqueue failed: %v", err) | ||||
| 	} | ||||
| 	_, err := client.Enqueue(task, TaskID(taskID)) | ||||
| 	if !errors.Is(err, ErrTaskIDConflict) { | ||||
| 		t.Errorf("Second task: Enqueue returned %v, want %v", err, ErrTaskIDConflict) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"}) | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	tests := []struct { | ||||
| @@ -398,7 +585,7 @@ func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 		task          *Task | ||||
| 		delay         time.Duration // value for ProcessIn option | ||||
| 		opts          []Option      // other options | ||||
| 		wantRes       *Result | ||||
| 		wantInfo      *TaskInfo | ||||
| 		wantPending   map[string][]*base.TaskMessage | ||||
| 		wantScheduled map[string][]base.Z | ||||
| 	}{ | ||||
| @@ -407,12 +594,18 @@ func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 			task:  task, | ||||
| 			delay: 1 * time.Hour, | ||||
| 			opts:  []Option{}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now.Add(1 * time.Hour), | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStateScheduled, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: time.Now().Add(1 * time.Hour), | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": {}, | ||||
| @@ -421,8 +614,8 @@ func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Message: &base.TaskMessage{ | ||||
| 							Type:     task.Type, | ||||
| 							Payload:  task.Payload.data, | ||||
| 							Type:     task.Type(), | ||||
| 							Payload:  task.Payload(), | ||||
| 							Retry:    defaultMaxRetry, | ||||
| 							Queue:    "default", | ||||
| 							Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -438,18 +631,24 @@ func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 			task:  task, | ||||
| 			delay: 0, | ||||
| 			opts:  []Option{}, | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "default", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          task.Type(), | ||||
| 				Payload:       task.Payload(), | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": { | ||||
| 					{ | ||||
| 						Type:     task.Type, | ||||
| 						Payload:  task.Payload.data, | ||||
| 						Type:     task.Type(), | ||||
| 						Payload:  task.Payload(), | ||||
| 						Retry:    defaultMaxRetry, | ||||
| 						Queue:    "default", | ||||
| 						Timeout:  int64(defaultTimeout.Seconds()), | ||||
| @@ -467,24 +666,24 @@ func TestClientEnqueueWithProcessInOption(t *testing.T) { | ||||
| 		h.FlushDB(t, r) // clean up db before each test case. | ||||
|  | ||||
| 		opts := append(tc.opts, ProcessIn(tc.delay)) | ||||
| 		gotRes, err := client.Enqueue(tc.task, opts...) | ||||
| 		gotInfo, err := client.Enqueue(tc.task, opts...) | ||||
| 		if err != nil { | ||||
| 			t.Error(err) | ||||
| 			continue | ||||
| 		} | ||||
| 		cmpOptions := []cmp.Option{ | ||||
| 			cmpopts.IgnoreFields(Result{}, "ID", "EnqueuedAt"), | ||||
| 			cmpopts.IgnoreFields(TaskInfo{}, "ID"), | ||||
| 			cmpopts.EquateApproxTime(500 * time.Millisecond), | ||||
| 		} | ||||
| 		if diff := cmp.Diff(tc.wantRes, gotRes, cmpOptions...); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantInfo, gotInfo, cmpOptions...); diff != "" { | ||||
| 			t.Errorf("%s;\nEnqueue(task, ProcessIn(%v)) returned %v, want %v; (-want,+got)\n%s", | ||||
| 				tc.desc, tc.delay, gotRes, tc.wantRes, diff) | ||||
| 				tc.desc, tc.delay, gotInfo, tc.wantInfo, diff) | ||||
| 		} | ||||
|  | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			gotPending := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotPending, h.IgnoreIDOpt, cmpopts.EquateEmpty()); diff != "" { | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.QueueKey(qname), diff) | ||||
| 				t.Errorf("%s;\nmismatch found in %q; (-want,+got)\n%s", tc.desc, base.PendingKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 		for qname, want := range tc.wantScheduled { | ||||
| @@ -501,7 +700,7 @@ func TestClientEnqueueError(t *testing.T) { | ||||
| 	client := NewClient(getRedisConnOpt(t)) | ||||
| 	defer client.Close() | ||||
|  | ||||
| 	task := NewTask("send_email", map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"}) | ||||
| 	task := NewTask("send_email", h.JSON(map[string]interface{}{"to": "customer@gmail.com", "from": "merchant@example.com"})) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| @@ -515,6 +714,26 @@ func TestClientEnqueueError(t *testing.T) { | ||||
| 				Queue(""), | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With empty task typename", | ||||
| 			task: NewTask("", h.JSON(map[string]interface{}{})), | ||||
| 			opts: []Option{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With blank task typename", | ||||
| 			task: NewTask("    ", h.JSON(map[string]interface{}{})), | ||||
| 			opts: []Option{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With empty task ID", | ||||
| 			task: NewTask("foo", nil), | ||||
| 			opts: []Option{TaskID("")}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With blank task ID", | ||||
| 			task: NewTask("foo", nil), | ||||
| 			opts: []Option{TaskID("  ")}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| @@ -527,17 +746,18 @@ func TestClientEnqueueError(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestClientDefaultOptions(t *testing.T) { | ||||
| func TestClientWithDefaultOptions(t *testing.T) { | ||||
| 	r := setup(t) | ||||
|  | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc        string | ||||
| 		defaultOpts []Option // options set at the client level. | ||||
| 		defaultOpts []Option // options set at task initialization time | ||||
| 		opts        []Option // options used at enqueue time. | ||||
| 		task        *Task | ||||
| 		wantRes     *Result | ||||
| 		tasktype    string | ||||
| 		payload     []byte | ||||
| 		wantInfo    *TaskInfo | ||||
| 		queue       string // queue that the message should go into. | ||||
| 		want        *base.TaskMessage | ||||
| 	}{ | ||||
| @@ -545,13 +765,20 @@ func TestClientDefaultOptions(t *testing.T) { | ||||
| 			desc:        "With queue routing option", | ||||
| 			defaultOpts: []Option{Queue("feed")}, | ||||
| 			opts:        []Option{}, | ||||
| 			task:        NewTask("feed:import", nil), | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			tasktype:    "feed:import", | ||||
| 			payload:     nil, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "feed", | ||||
| 				Retry:     defaultMaxRetry, | ||||
| 				Type:          "feed:import", | ||||
| 				Payload:       nil, | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      defaultMaxRetry, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			queue: "feed", | ||||
| 			want: &base.TaskMessage{ | ||||
| @@ -567,13 +794,20 @@ func TestClientDefaultOptions(t *testing.T) { | ||||
| 			desc:        "With multiple options", | ||||
| 			defaultOpts: []Option{Queue("feed"), MaxRetry(5)}, | ||||
| 			opts:        []Option{}, | ||||
| 			task:        NewTask("feed:import", nil), | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			tasktype:    "feed:import", | ||||
| 			payload:     nil, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "feed", | ||||
| 				Retry:     5, | ||||
| 				Type:          "feed:import", | ||||
| 				Payload:       nil, | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      5, | ||||
| 				Retried:       0, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			queue: "feed", | ||||
| 			want: &base.TaskMessage{ | ||||
| @@ -589,13 +823,19 @@ func TestClientDefaultOptions(t *testing.T) { | ||||
| 			desc:        "With overriding options at enqueue time", | ||||
| 			defaultOpts: []Option{Queue("feed"), MaxRetry(5)}, | ||||
| 			opts:        []Option{Queue("critical")}, | ||||
| 			task:        NewTask("feed:import", nil), | ||||
| 			wantRes: &Result{ | ||||
| 				ProcessAt: now, | ||||
| 			tasktype:    "feed:import", | ||||
| 			payload:     nil, | ||||
| 			wantInfo: &TaskInfo{ | ||||
| 				Queue:         "critical", | ||||
| 				Retry:     5, | ||||
| 				Type:          "feed:import", | ||||
| 				Payload:       nil, | ||||
| 				State:         TaskStatePending, | ||||
| 				MaxRetry:      5, | ||||
| 				LastErr:       "", | ||||
| 				LastFailedAt:  time.Time{}, | ||||
| 				Timeout:       defaultTimeout, | ||||
| 				Deadline:  noDeadline, | ||||
| 				Deadline:      time.Time{}, | ||||
| 				NextProcessAt: now, | ||||
| 			}, | ||||
| 			queue: "critical", | ||||
| 			want: &base.TaskMessage{ | ||||
| @@ -613,18 +853,18 @@ func TestClientDefaultOptions(t *testing.T) { | ||||
| 		h.FlushDB(t, r) | ||||
| 		c := NewClient(getRedisConnOpt(t)) | ||||
| 		defer c.Close() | ||||
| 		c.SetDefaultOptions(tc.task.Type, tc.defaultOpts...) | ||||
| 		gotRes, err := c.Enqueue(tc.task, tc.opts...) | ||||
| 		task := NewTask(tc.tasktype, tc.payload, tc.defaultOpts...) | ||||
| 		gotInfo, err := c.Enqueue(task, tc.opts...) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		cmpOptions := []cmp.Option{ | ||||
| 			cmpopts.IgnoreFields(Result{}, "ID", "EnqueuedAt"), | ||||
| 			cmpopts.IgnoreFields(TaskInfo{}, "ID"), | ||||
| 			cmpopts.EquateApproxTime(500 * time.Millisecond), | ||||
| 		} | ||||
| 		if diff := cmp.Diff(tc.wantRes, gotRes, cmpOptions...); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantInfo, gotInfo, cmpOptions...); diff != "" { | ||||
| 			t.Errorf("%s;\nEnqueue(task, opts...) returned %v, want %v; (-want,+got)\n%s", | ||||
| 				tc.desc, gotRes, tc.wantRes, diff) | ||||
| 				tc.desc, gotInfo, tc.wantInfo, diff) | ||||
| 		} | ||||
| 		pending := h.GetPendingMessages(t, r, tc.queue) | ||||
| 		if len(pending) != 1 { | ||||
| @@ -650,7 +890,7 @@ func TestClientEnqueueUnique(t *testing.T) { | ||||
| 		ttl  time.Duration | ||||
| 	}{ | ||||
| 		{ | ||||
| 			NewTask("email", map[string]interface{}{"user_id": 123}), | ||||
| 			NewTask("email", h.JSON(map[string]interface{}{"user_id": 123})), | ||||
| 			time.Hour, | ||||
| 		}, | ||||
| 	} | ||||
| @@ -664,7 +904,7 @@ func TestClientEnqueueUnique(t *testing.T) { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		gotTTL := r.TTL(base.UniqueKey(base.DefaultQueueName, tc.task.Type, tc.task.Payload.data)).Val() | ||||
| 		gotTTL := r.TTL(context.Background(), base.UniqueKey(base.DefaultQueueName, tc.task.Type(), tc.task.Payload())).Val() | ||||
| 		if !cmp.Equal(tc.ttl.Seconds(), gotTTL.Seconds(), cmpopts.EquateApprox(0, 1)) { | ||||
| 			t.Errorf("TTL = %v, want %v", gotTTL, tc.ttl) | ||||
| 			continue | ||||
| @@ -709,7 +949,7 @@ func TestClientEnqueueUniqueWithProcessInOption(t *testing.T) { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		gotTTL := r.TTL(base.UniqueKey(base.DefaultQueueName, tc.task.Type, tc.task.Payload.data)).Val() | ||||
| 		gotTTL := r.TTL(context.Background(), base.UniqueKey(base.DefaultQueueName, tc.task.Type(), tc.task.Payload())).Val() | ||||
| 		wantTTL := time.Duration(tc.ttl.Seconds()+tc.d.Seconds()) * time.Second | ||||
| 		if !cmp.Equal(wantTTL.Seconds(), gotTTL.Seconds(), cmpopts.EquateApprox(0, 1)) { | ||||
| 			t.Errorf("TTL = %v, want %v", gotTTL, wantTTL) | ||||
| @@ -755,7 +995,7 @@ func TestClientEnqueueUniqueWithProcessAtOption(t *testing.T) { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		gotTTL := r.TTL(base.UniqueKey(base.DefaultQueueName, tc.task.Type, tc.task.Payload.data)).Val() | ||||
| 		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()) | ||||
| 		if !cmp.Equal(wantTTL.Seconds(), gotTTL.Seconds(), cmpopts.EquateApprox(0, 1)) { | ||||
| 			t.Errorf("TTL = %v, want %v", gotTTL, wantTTL) | ||||
| @@ -774,71 +1014,3 @@ func TestClientEnqueueUniqueWithProcessAtOption(t *testing.T) { | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestParseOption(t *testing.T) { | ||||
| 	oneHourFromNow := time.Now().Add(1 * time.Hour) | ||||
| 	tests := []struct { | ||||
| 		s        string | ||||
| 		wantType OptionType | ||||
| 		wantVal  interface{} | ||||
| 	}{ | ||||
| 		{`MaxRetry(10)`, MaxRetryOpt, 10}, | ||||
| 		{`Queue("email")`, QueueOpt, "email"}, | ||||
| 		{`Timeout(3m)`, TimeoutOpt, 3 * time.Minute}, | ||||
| 		{Deadline(oneHourFromNow).String(), DeadlineOpt, oneHourFromNow}, | ||||
| 		{`Unique(1h)`, UniqueOpt, 1 * time.Hour}, | ||||
| 		{ProcessAt(oneHourFromNow).String(), ProcessAtOpt, oneHourFromNow}, | ||||
| 		{`ProcessIn(10m)`, ProcessInOpt, 10 * time.Minute}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		t.Run(tc.s, func(t *testing.T) { | ||||
| 			got, err := parseOption(tc.s) | ||||
| 			if err != nil { | ||||
| 				t.Fatalf("returned error: %v", err) | ||||
| 			} | ||||
| 			if got == nil { | ||||
| 				t.Fatal("returned nil") | ||||
| 			} | ||||
| 			if got.Type() != tc.wantType { | ||||
| 				t.Fatalf("got type %v, want type %v ", got.Type(), tc.wantType) | ||||
| 			} | ||||
| 			switch tc.wantType { | ||||
| 			case QueueOpt: | ||||
| 				gotVal, ok := got.Value().(string) | ||||
| 				if !ok { | ||||
| 					t.Fatal("returned Option with non-string value") | ||||
| 				} | ||||
| 				if gotVal != tc.wantVal.(string) { | ||||
| 					t.Fatalf("got value %v, want %v", gotVal, tc.wantVal) | ||||
| 				} | ||||
| 			case MaxRetryOpt: | ||||
| 				gotVal, ok := got.Value().(int) | ||||
| 				if !ok { | ||||
| 					t.Fatal("returned Option with non-int value") | ||||
| 				} | ||||
| 				if gotVal != tc.wantVal.(int) { | ||||
| 					t.Fatalf("got value %v, want %v", gotVal, tc.wantVal) | ||||
| 				} | ||||
| 			case TimeoutOpt, UniqueOpt, ProcessInOpt: | ||||
| 				gotVal, ok := got.Value().(time.Duration) | ||||
| 				if !ok { | ||||
| 					t.Fatal("returned Option with non duration value") | ||||
| 				} | ||||
| 				if gotVal != tc.wantVal.(time.Duration) { | ||||
| 					t.Fatalf("got value %v, want %v", gotVal, tc.wantVal) | ||||
| 				} | ||||
| 			case DeadlineOpt, ProcessAtOpt: | ||||
| 				gotVal, ok := got.Value().(time.Time) | ||||
| 				if !ok { | ||||
| 					t.Fatal("returned Option with non time value") | ||||
| 				} | ||||
| 				if cmp.Equal(gotVal, tc.wantVal.(time.Time)) { | ||||
| 					t.Fatalf("got value %v, want %v", gotVal, tc.wantVal) | ||||
| 				} | ||||
| 			default: | ||||
| 				t.Fatalf("returned Option with unexpected type: %v", got.Type()) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										55
									
								
								context.go
									
									
									
									
									
								
							
							
						
						
									
										55
									
								
								context.go
									
									
									
									
									
								
							| @@ -6,49 +6,16 @@ package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| ) | ||||
|  | ||||
| // A taskMetadata holds task scoped data to put in context. | ||||
| type taskMetadata struct { | ||||
| 	id         string | ||||
| 	maxRetry   int | ||||
| 	retryCount int | ||||
| 	qname      string | ||||
| } | ||||
|  | ||||
| // ctxKey type is unexported to prevent collisions with context keys defined in | ||||
| // other packages. | ||||
| type ctxKey int | ||||
|  | ||||
| // metadataCtxKey is the context key for the task metadata. | ||||
| // Its value of zero is arbitrary. | ||||
| const metadataCtxKey ctxKey = 0 | ||||
|  | ||||
| // createContext returns a context and cancel function for a given task message. | ||||
| func createContext(msg *base.TaskMessage, deadline time.Time) (context.Context, context.CancelFunc) { | ||||
| 	metadata := taskMetadata{ | ||||
| 		id:         msg.ID.String(), | ||||
| 		maxRetry:   msg.Retry, | ||||
| 		retryCount: msg.Retried, | ||||
| 		qname:      msg.Queue, | ||||
| 	} | ||||
| 	ctx := context.WithValue(context.Background(), metadataCtxKey, metadata) | ||||
| 	return context.WithDeadline(ctx, deadline) | ||||
| } | ||||
|  | ||||
| // GetTaskID extracts a task ID from a context, if any. | ||||
| // | ||||
| // ID of a task is guaranteed to be unique. | ||||
| // ID of a task doesn't change if the task is being retried. | ||||
| func GetTaskID(ctx context.Context) (id string, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return "", false | ||||
| 	} | ||||
| 	return metadata.id, true | ||||
| 	return asynqcontext.GetTaskID(ctx) | ||||
| } | ||||
|  | ||||
| // GetRetryCount extracts retry count from a context, if any. | ||||
| @@ -56,11 +23,7 @@ func GetTaskID(ctx context.Context) (id string, ok bool) { | ||||
| // Return value n indicates the number of times associated task has been | ||||
| // retried so far. | ||||
| func GetRetryCount(ctx context.Context) (n int, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return 0, false | ||||
| 	} | ||||
| 	return metadata.retryCount, true | ||||
| 	return asynqcontext.GetRetryCount(ctx) | ||||
| } | ||||
|  | ||||
| // GetMaxRetry extracts maximum retry from a context, if any. | ||||
| @@ -68,20 +31,12 @@ func GetRetryCount(ctx context.Context) (n int, ok bool) { | ||||
| // Return value n indicates the maximum number of times the assoicated task | ||||
| // can be retried if ProcessTask returns a non-nil error. | ||||
| func GetMaxRetry(ctx context.Context) (n int, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return 0, false | ||||
| 	} | ||||
| 	return metadata.maxRetry, true | ||||
| 	return asynqcontext.GetMaxRetry(ctx) | ||||
| } | ||||
|  | ||||
| // GetQueueName extracts queue name from a context, if any. | ||||
| // | ||||
| // Return value qname indicates which queue the task was pulled from. | ||||
| func GetQueueName(ctx context.Context) (qname string, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return "", false | ||||
| 	} | ||||
| 	return metadata.qname, true | ||||
| 	return asynqcontext.GetQueueName(ctx) | ||||
| } | ||||
|   | ||||
							
								
								
									
										27
									
								
								doc.go
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								doc.go
									
									
									
									
									
								
							| @@ -11,7 +11,7 @@ specify the connection using one of RedisConnOpt types. | ||||
|     redisConnOpt = asynq.RedisClientOpt{ | ||||
|         Addr:     "127.0.0.1:6379", | ||||
|         Password: "xxxxx", | ||||
|         DB:       3, | ||||
|         DB:       2, | ||||
|     } | ||||
|  | ||||
| The Client is used to enqueue a task. | ||||
| @@ -20,15 +20,19 @@ The Client is used to enqueue a task. | ||||
|     client := asynq.NewClient(redisConnOpt) | ||||
|  | ||||
|     // Task is created with two parameters: its type and payload. | ||||
|     t := asynq.NewTask( | ||||
|         "send_email", | ||||
|         map[string]interface{}{"user_id": 42}) | ||||
|     // Payload data is simply an array of bytes. It can be encoded in JSON, Protocol Buffer, Gob, etc. | ||||
|     b, err := json.Marshal(ExamplePayload{UserID: 42}) | ||||
|     if err != nil { | ||||
|         log.Fatal(err) | ||||
|     } | ||||
|  | ||||
|     task := asynq.NewTask("example", b) | ||||
|  | ||||
|     // Enqueue the task to be processed immediately. | ||||
|     res, err := client.Enqueue(t) | ||||
|     info, err := client.Enqueue(task) | ||||
|  | ||||
|     // Schedule the task to be processed after one minute. | ||||
|     res, err = client.Enqueue(t, asynq.ProcessIn(1*time.Minute)) | ||||
|     info, err = client.Enqueue(t, asynq.ProcessIn(1*time.Minute)) | ||||
|  | ||||
| The Server is used to run the task processing workers with a given | ||||
| handler. | ||||
| @@ -52,10 +56,13 @@ Example of a type that implements the Handler interface. | ||||
|  | ||||
|     func (h *TaskHandler) ProcessTask(ctx context.Context, task *asynq.Task) error { | ||||
|         switch task.Type { | ||||
|         case "send_email": | ||||
|             id, err := task.Payload.GetInt("user_id") | ||||
|             // send email | ||||
|         //... | ||||
|         case "example": | ||||
|             var data ExamplePayload | ||||
|             if err := json.Unmarshal(task.Payload(), &data); err != nil { | ||||
|                 return err | ||||
|             } | ||||
|             // perform task with the data | ||||
|  | ||||
|         default: | ||||
|             return fmt.Errorf("unexpected task type %q", task.Type) | ||||
|         } | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								docs/assets/asynqmon-queues-view.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/assets/asynqmon-queues-view.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 279 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/asynqmon-task-view.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								docs/assets/asynqmon-task-view.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 347 KiB | 
| @@ -30,7 +30,7 @@ func ExampleServer_Run() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ExampleServer_Stop() { | ||||
| func ExampleServer_Shutdown() { | ||||
| 	srv := asynq.NewServer( | ||||
| 		asynq.RedisClientOpt{Addr: ":6379"}, | ||||
| 		asynq.Config{Concurrency: 20}, | ||||
| @@ -47,10 +47,10 @@ func ExampleServer_Stop() { | ||||
| 	signal.Notify(sigs, unix.SIGTERM, unix.SIGINT) | ||||
| 	<-sigs // wait for termination signal | ||||
|  | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func ExampleServer_Quiet() { | ||||
| func ExampleServer_Stop() { | ||||
| 	srv := asynq.NewServer( | ||||
| 		asynq.RedisClientOpt{Addr: ":6379"}, | ||||
| 		asynq.Config{Concurrency: 20}, | ||||
| @@ -70,13 +70,13 @@ func ExampleServer_Quiet() { | ||||
| 	for { | ||||
| 		s := <-sigs | ||||
| 		if s == unix.SIGTSTP { | ||||
| 			srv.Quiet() // stop processing new tasks | ||||
| 			srv.Stop() // stop processing new tasks | ||||
| 			continue | ||||
| 		} | ||||
| 		break | ||||
| 		break // received SIGTERM or SIGINT signal | ||||
| 	} | ||||
|  | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func ExampleScheduler() { | ||||
|   | ||||
| @@ -45,7 +45,7 @@ func newForwarder(params forwarderParams) *forwarder { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (f *forwarder) terminate() { | ||||
| func (f *forwarder) shutdown() { | ||||
| 	f.logger.Debug("Forwarder shutting down...") | ||||
| 	// Signal the forwarder goroutine to stop polling. | ||||
| 	f.done <- struct{}{} | ||||
| @@ -69,7 +69,7 @@ func (f *forwarder) start(wg *sync.WaitGroup) { | ||||
| } | ||||
|  | ||||
| func (f *forwarder) exec() { | ||||
| 	if err := f.broker.CheckAndEnqueue(f.queues...); err != nil { | ||||
| 	if err := f.broker.ForwardIfReady(f.queues...); err != nil { | ||||
| 		f.logger.Errorf("Could not enqueue scheduled tasks: %v", err) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -111,7 +111,7 @@ func TestForwarder(t *testing.T) { | ||||
| 		var wg sync.WaitGroup | ||||
| 		s.start(&wg) | ||||
| 		time.Sleep(tc.wait) | ||||
| 		s.terminate() | ||||
| 		s.shutdown() | ||||
|  | ||||
| 		for qname, want := range tc.wantScheduled { | ||||
| 			gotScheduled := h.GetScheduledMessages(t, r, qname) | ||||
| @@ -130,7 +130,7 @@ func TestForwarder(t *testing.T) { | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			gotPending := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotPending, h.SortMsgOpt); diff != "" { | ||||
| 				t.Errorf("mismatch found in %q after running forwarder: (-want, +got)\n%s", base.QueueKey(qname), diff) | ||||
| 				t.Errorf("mismatch found in %q after running forwarder: (-want, +got)\n%s", base.PendingKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										14
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								go.mod
									
									
									
									
									
								
							| @@ -3,13 +3,17 @@ module github.com/hibiken/asynq | ||||
| go 1.13 | ||||
|  | ||||
| require ( | ||||
| 	github.com/go-redis/redis/v7 v7.4.0 | ||||
| 	github.com/google/go-cmp v0.4.0 | ||||
| 	github.com/google/uuid v1.1.1 | ||||
| 	github.com/go-redis/redis/v8 v8.11.2 | ||||
| 	github.com/golang/protobuf v1.4.2 | ||||
| 	github.com/google/go-cmp v0.5.6 | ||||
| 	github.com/google/uuid v1.2.0 | ||||
| 	github.com/kr/pretty v0.1.0 // indirect | ||||
| 	github.com/robfig/cron/v3 v3.0.1 | ||||
| 	github.com/spf13/cast v1.3.1 | ||||
| 	github.com/stretchr/testify v1.6.1 // indirect | ||||
| 	go.uber.org/goleak v0.10.0 | ||||
| 	golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e | ||||
| 	golang.org/x/sys v0.0.0-20210112080510-489259a85091 | ||||
| 	golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 | ||||
| 	gopkg.in/yaml.v2 v2.2.7 // indirect | ||||
| 	google.golang.org/protobuf v1.25.0 | ||||
| 	gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										165
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										165
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,61 +1,165 @@ | ||||
| cloud.google.com/go v0.26.0 h1:e0WKqKTd5BnrG8aKH3J3h+QvEIQtSUcf2n5UZ5ZgLtQ= | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1 h1:glEXhBS5PSLLv4IXzLA5yPRVX4bilULVyxxbrfOtDAk= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/client9/misspell v0.3.4 h1:ta993UF76GwbvJcIo3Y68y/M3WxlpEHPWIGDkJYwzJI= | ||||
| 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/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I= | ||||
| 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 h1:4cmBvAEBNJaGARUEs3/suWRyfyBfhf7I60WBZq+bv2w= | ||||
| 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 h1:EQciDnbrYxy13PgWoY8AqoxGiPrpgBZ1R8UNe3ddc+A= | ||||
| 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/go-redis/redis/v7 v7.2.0 h1:CrCexy/jYWZjW0AyVoHlcJUeZN19VWlbepTh1Vq6dJs= | ||||
| github.com/go-redis/redis/v7 v7.2.0/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRfnwdHj/Dcg= | ||||
| github.com/go-redis/redis/v7 v7.4.0 h1:7obg6wUoj05T0EpY0o8B59S9w5yeMWql7sw2kwNW1x4= | ||||
| github.com/go-redis/redis/v7 v7.4.0/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRfnwdHj/Dcg= | ||||
| 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-redis/redis/v8 v8.11.2 h1:WqlSpAwz8mxDSMCvbyz1Mkiqe0LE5OY4j3lgkvu1Ts0= | ||||
| github.com/go-redis/redis/v8 v8.11.2/go.mod h1:DLomh7y2e3ggQXQLd1YgmvIfecPJoFl7WU5SOQ/r06M= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/mock v1.1.1 h1:G5FRp8JnTd7RQH5kemVNlMeyXQAztQ3mOWV95KxsXH8= | ||||
| 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.2 h1:6nsPYzhq5kReh6QImI3k5qWzO4PEbvbIW2cwSfR/6xs= | ||||
| github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/google/go-cmp v0.4.0 h1:xsAVV57WRhGj6kEIi8ReJzQlHHqcBYCElAvkovg3B/4= | ||||
| 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 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| 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/uuid v1.1.1 h1:Gkbcsh/GbpXz7lPftLA3P6TYMwjCLYm83jiFQZF/3gY= | ||||
| github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/go-cmp v0.5.0/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/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| 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 h1:VkoXIwSboBpnk99O/KFauAEILuNHv5DVFKZMBN/gUgw= | ||||
| 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/nxadm/tail v1.4.4 h1:DQuhQpB1tVlglWS2hLQ5OV6B5r8aGxSrPc5Qo6uTN78= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.10.1 h1:q/mM8GF/n0shIN8SaAZ0V+jnLPzen6WIVZdiwrRlMlo= | ||||
| github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/gomega v1.7.0 h1:XPnZz8VVBHjVsy1vzJmRwIcSwiUO+JFfrv/xGiigmME= | ||||
| github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.15.0 h1:1V1NfVQR87RtWAgp1lv9JZJ5Jap+XFGKPi00andXGi4= | ||||
| github.com/onsi/ginkgo v1.15.0/go.mod h1:hF8qUzuuC8DJGygJH3726JnCZX4MYbRB8yFfISqnKUg= | ||||
| 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 h1:7n6FEkpFmfCoo2t+YYqXH0evK+a9ICQz0xcAy9dYcaQ= | ||||
| github.com/onsi/gomega v1.10.5/go.mod h1:gza4q3jKQJijlu05nKWRCW/GavJumGt8aNRxWg7mt48= | ||||
| 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_model v0.0.0-20190812154241-14fe0d1b01d4 h1:gQz4mCbXsO+nc9n1hCxHcGA3Zx3Eo+UHZoInFGUIXNM= | ||||
| github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
| 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/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= | ||||
| github.com/stretchr/objx v0.1.0 h1:4G4v2dO3VZwixGIRoQ5Lfboy6nUhCyYzaqnIAPPhYs4= | ||||
| 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.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 h1:ruQGxdhGHe7FWOJPT0mKs5+pD2Xs1Bm/kdGlHO04FmM= | ||||
| 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-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd h1:nTDtHvHSdCn1m6ITfMRqtOd/9+7a3s8RBNOZ3eYZzJA= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4 h1:c2HOrn5iMezYjSlGPncknSEr/8x5LELb/ilJbXi9DEA= | ||||
| 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 h1:XQyxROzUlZH+WIQwySDgnISgOivlhjIEwaQaJEJrrN0= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mod v0.3.0 h1:RM4zey1++hCTbCVQfnWeKs9/IEsaBLA8vTkd0WVtmH4= | ||||
| 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-20190923162816-aa69164e4478 h1:l5EDrHhldLYb3ZRHDUhXF7Om7MvYXnkV9/iQNo1lX6g= | ||||
| golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| 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-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-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb h1:eBmm0M9fYhWpKZLjQUUKka/LtIxf46G4fxeEz5KJr9U= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be h1:vEDujvNQGv4jgYKudGeI/+DAX4Jffq6hpD55MmoEvKs= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e h1:o3PsSEY8E4eXWkXrIP9YJALUkVZqzHJT5DOasTyn8Vs= | ||||
| golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/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-20201020160332-67f06af15bc9 h1:SQFwaSi55rU7vdNs9Yr0Z324VNlrF+0wMqRXT4St8ck= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/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-20180909124046-d0be0721c37e/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-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e h1:9vRrk9YW2BTzLP0VCB9ZDjU4cPqkg+IDWL7XgxA1yxQ= | ||||
| golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/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-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091 h1:DMyOG0U+gKfu8JZzg2UQe9MeaC1X+xQWlAKcRnjxjCw= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k= | ||||
| 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/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= | ||||
| 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 h1:4nW4NLDYnU28ojHaHO8OVxFHk/aQ33U01a9cjED+pzE= | ||||
| 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 h1:/wp5JvzpHIxhs/dumFmF7BXTf3Z+dd4uXta4kVyO508= | ||||
| 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 h1:+kGHl1aib/qcwaRi1CbqBZ1rk19r85MNUf8HaBghugY= | ||||
| 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 h1:rRYRFMVgRv6E0D70Skyfsr28tDXIuuPZyWGMPdMcnXg= | ||||
| 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 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/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= | ||||
| @@ -63,8 +167,11 @@ gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= | ||||
| 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 h1:mUhvW9EsL+naU5Q3cakzfE91YhliOondGd6ZrsDBHQE= | ||||
| gopkg.in/yaml.v2 v2.2.1/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.7 h1:VUgggvou5XRW9mHwD/yXxIYSMtY0zoKQf/v226p2nyo= | ||||
| gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| 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 h1:/hemPrYIhOhy8zYrNj+069zDB68us2sMGsfkFJO0iZs= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
|   | ||||
| @@ -45,7 +45,7 @@ func newHealthChecker(params healthcheckerParams) *healthchecker { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (hc *healthchecker) terminate() { | ||||
| func (hc *healthchecker) shutdown() { | ||||
| 	if hc.healthcheckFunc == nil { | ||||
| 		return | ||||
| 	} | ||||
|   | ||||
| @@ -51,7 +51,7 @@ func TestHealthChecker(t *testing.T) { | ||||
| 	} | ||||
| 	mu.Unlock() | ||||
|  | ||||
| 	hc.terminate() | ||||
| 	hc.shutdown() | ||||
| } | ||||
|  | ||||
| func TestHealthCheckerWhenRedisDown(t *testing.T) { | ||||
| @@ -99,5 +99,5 @@ func TestHealthCheckerWhenRedisDown(t *testing.T) { | ||||
| 	} | ||||
| 	mu.Unlock() | ||||
|  | ||||
| 	hc.terminate() | ||||
| 	hc.shutdown() | ||||
| } | ||||
|   | ||||
							
								
								
									
										48
									
								
								heartbeat.go
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								heartbeat.go
									
									
									
									
									
								
							| @@ -38,13 +38,13 @@ type heartbeater struct { | ||||
| 	// heartbeater goroutine. In other words, confine these variables | ||||
| 	// to this goroutine only. | ||||
| 	started time.Time | ||||
| 	workers map[string]workerStat | ||||
| 	workers map[string]*workerInfo | ||||
|  | ||||
| 	// status is shared with other goroutine but is concurrency safe. | ||||
| 	status *base.ServerStatus | ||||
| 	// state is shared with other goroutine but is concurrency safe. | ||||
| 	state *base.ServerState | ||||
|  | ||||
| 	// channels to receive updates on active workers. | ||||
| 	starting <-chan *base.TaskMessage | ||||
| 	starting <-chan *workerInfo | ||||
| 	finished <-chan *base.TaskMessage | ||||
| } | ||||
|  | ||||
| @@ -55,8 +55,8 @@ type heartbeaterParams struct { | ||||
| 	concurrency    int | ||||
| 	queues         map[string]int | ||||
| 	strictPriority bool | ||||
| 	status         *base.ServerStatus | ||||
| 	starting       <-chan *base.TaskMessage | ||||
| 	state          *base.ServerState | ||||
| 	starting       <-chan *workerInfo | ||||
| 	finished       <-chan *base.TaskMessage | ||||
| } | ||||
|  | ||||
| @@ -79,24 +79,27 @@ func newHeartbeater(params heartbeaterParams) *heartbeater { | ||||
| 		queues:         params.queues, | ||||
| 		strictPriority: params.strictPriority, | ||||
|  | ||||
| 		status:   params.status, | ||||
| 		workers:  make(map[string]workerStat), | ||||
| 		state:    params.state, | ||||
| 		workers:  make(map[string]*workerInfo), | ||||
| 		starting: params.starting, | ||||
| 		finished: params.finished, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *heartbeater) terminate() { | ||||
| func (h *heartbeater) shutdown() { | ||||
| 	h.logger.Debug("Heartbeater shutting down...") | ||||
| 	// Signal the heartbeater goroutine to stop. | ||||
| 	h.done <- struct{}{} | ||||
| } | ||||
|  | ||||
| // A workerStat records the message a worker is working on | ||||
| // and the time the worker has started processing the message. | ||||
| type workerStat struct { | ||||
| 	started time.Time | ||||
| // A workerInfo holds an active worker information. | ||||
| type workerInfo struct { | ||||
| 	// the task message the worker is processing. | ||||
| 	msg *base.TaskMessage | ||||
| 	// the time the worker has started processing the message. | ||||
| 	started time.Time | ||||
| 	// deadline the worker has to finish processing the task by. | ||||
| 	deadline time.Time | ||||
| } | ||||
|  | ||||
| func (h *heartbeater) start(wg *sync.WaitGroup) { | ||||
| @@ -121,11 +124,11 @@ func (h *heartbeater) start(wg *sync.WaitGroup) { | ||||
| 				h.beat() | ||||
| 				timer.Reset(h.interval) | ||||
|  | ||||
| 			case msg := <-h.starting: | ||||
| 				h.workers[msg.ID.String()] = workerStat{time.Now(), msg} | ||||
| 			case w := <-h.starting: | ||||
| 				h.workers[w.msg.ID] = w | ||||
|  | ||||
| 			case msg := <-h.finished: | ||||
| 				delete(h.workers, msg.ID.String()) | ||||
| 				delete(h.workers, msg.ID) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| @@ -139,22 +142,23 @@ func (h *heartbeater) beat() { | ||||
| 		Concurrency:       h.concurrency, | ||||
| 		Queues:            h.queues, | ||||
| 		StrictPriority:    h.strictPriority, | ||||
| 		Status:            h.status.String(), | ||||
| 		Status:            h.state.String(), | ||||
| 		Started:           h.started, | ||||
| 		ActiveWorkerCount: len(h.workers), | ||||
| 	} | ||||
|  | ||||
| 	var ws []*base.WorkerInfo | ||||
| 	for id, stat := range h.workers { | ||||
| 	for id, w := range h.workers { | ||||
| 		ws = append(ws, &base.WorkerInfo{ | ||||
| 			Host:     h.host, | ||||
| 			PID:      h.pid, | ||||
| 			ServerID: h.serverID, | ||||
| 			ID:       id, | ||||
| 			Type:     stat.msg.Type, | ||||
| 			Queue:    stat.msg.Queue, | ||||
| 			Payload:  stat.msg.Payload, | ||||
| 			Started:  stat.started, | ||||
| 			Type:     w.msg.Type, | ||||
| 			Queue:    w.msg.Queue, | ||||
| 			Payload:  w.msg.Payload, | ||||
| 			Started:  w.started, | ||||
| 			Deadline: w.deadline, | ||||
| 		}) | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -38,7 +38,7 @@ func TestHeartbeater(t *testing.T) { | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r) | ||||
|  | ||||
| 		status := base.NewServerStatus(base.StatusIdle) | ||||
| 		state := base.NewServerState() | ||||
| 		hb := newHeartbeater(heartbeaterParams{ | ||||
| 			logger:         testLogger, | ||||
| 			broker:         rdbClient, | ||||
| @@ -46,8 +46,8 @@ func TestHeartbeater(t *testing.T) { | ||||
| 			concurrency:    tc.concurrency, | ||||
| 			queues:         tc.queues, | ||||
| 			strictPriority: false, | ||||
| 			status:         status, | ||||
| 			starting:       make(chan *base.TaskMessage), | ||||
| 			state:          state, | ||||
| 			starting:       make(chan *workerInfo), | ||||
| 			finished:       make(chan *base.TaskMessage), | ||||
| 		}) | ||||
|  | ||||
| @@ -55,7 +55,7 @@ func TestHeartbeater(t *testing.T) { | ||||
| 		hb.host = tc.host | ||||
| 		hb.pid = tc.pid | ||||
|  | ||||
| 		status.Set(base.StatusRunning) | ||||
| 		state.Set(base.StateActive) | ||||
| 		var wg sync.WaitGroup | ||||
| 		hb.start(&wg) | ||||
|  | ||||
| @@ -65,7 +65,7 @@ func TestHeartbeater(t *testing.T) { | ||||
| 			Queues:      tc.queues, | ||||
| 			Concurrency: tc.concurrency, | ||||
| 			Started:     time.Now(), | ||||
| 			Status:      "running", | ||||
| 			Status:      "active", | ||||
| 		} | ||||
|  | ||||
| 		// allow for heartbeater to write to redis | ||||
| @@ -74,49 +74,49 @@ func TestHeartbeater(t *testing.T) { | ||||
| 		ss, err := rdbClient.ListServers() | ||||
| 		if err != nil { | ||||
| 			t.Errorf("could not read server info from redis: %v", err) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if len(ss) != 1 { | ||||
| 			t.Errorf("(*RDB).ListServers returned %d process info, want 1", len(ss)) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if diff := cmp.Diff(want, ss[0], timeCmpOpt, ignoreOpt, ignoreFieldOpt); diff != "" { | ||||
| 			t.Errorf("redis stored process status %+v, want %+v; (-want, +got)\n%s", ss[0], want, diff) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// status change | ||||
| 		status.Set(base.StatusStopped) | ||||
| 		state.Set(base.StateClosed) | ||||
|  | ||||
| 		// allow for heartbeater to write to redis | ||||
| 		time.Sleep(tc.interval * 2) | ||||
|  | ||||
| 		want.Status = "stopped" | ||||
| 		want.Status = "closed" | ||||
| 		ss, err = rdbClient.ListServers() | ||||
| 		if err != nil { | ||||
| 			t.Errorf("could not read process status from redis: %v", err) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if len(ss) != 1 { | ||||
| 			t.Errorf("(*RDB).ListProcesses returned %d process info, want 1", len(ss)) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if diff := cmp.Diff(want, ss[0], timeCmpOpt, ignoreOpt, ignoreFieldOpt); diff != "" { | ||||
| 			t.Errorf("redis stored process status %+v, want %+v; (-want, +got)\n%s", ss[0], want, diff) | ||||
| 			hb.terminate() | ||||
| 			hb.shutdown() | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		hb.terminate() | ||||
| 		hb.shutdown() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -131,6 +131,8 @@ func TestHeartbeaterWithRedisDown(t *testing.T) { | ||||
| 	r := rdb.NewRDB(setup(t)) | ||||
| 	defer r.Close() | ||||
| 	testBroker := testbroker.NewTestBroker(r) | ||||
| 	state := base.NewServerState() | ||||
| 	state.Set(base.StateActive) | ||||
| 	hb := newHeartbeater(heartbeaterParams{ | ||||
| 		logger:         testLogger, | ||||
| 		broker:         testBroker, | ||||
| @@ -138,8 +140,8 @@ func TestHeartbeaterWithRedisDown(t *testing.T) { | ||||
| 		concurrency:    10, | ||||
| 		queues:         map[string]int{"default": 1}, | ||||
| 		strictPriority: false, | ||||
| 		status:         base.NewServerStatus(base.StatusRunning), | ||||
| 		starting:       make(chan *base.TaskMessage), | ||||
| 		state:          state, | ||||
| 		starting:       make(chan *workerInfo), | ||||
| 		finished:       make(chan *base.TaskMessage), | ||||
| 	}) | ||||
|  | ||||
| @@ -150,5 +152,5 @@ func TestHeartbeaterWithRedisDown(t *testing.T) { | ||||
| 	// wait for heartbeater to try writing data to redis | ||||
| 	time.Sleep(2 * time.Second) | ||||
|  | ||||
| 	hb.terminate() | ||||
| 	hb.shutdown() | ||||
| } | ||||
|   | ||||
							
								
								
									
										659
									
								
								inspector.go
									
									
									
									
									
								
							
							
						
						
									
										659
									
								
								inspector.go
									
									
									
									
									
								
							| @@ -10,7 +10,9 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| ) | ||||
|  | ||||
| @@ -20,10 +22,14 @@ type Inspector struct { | ||||
| 	rdb *rdb.RDB | ||||
| } | ||||
|  | ||||
| // NewInspector returns a new instance of Inspector. | ||||
| // New returns a new instance of Inspector. | ||||
| func NewInspector(r RedisConnOpt) *Inspector { | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("inspeq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	return &Inspector{ | ||||
| 		rdb: rdb.NewRDB(createRedisClient(r)), | ||||
| 		rdb: rdb.NewRDB(c), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -37,13 +43,19 @@ func (i *Inspector) Queues() ([]string, error) { | ||||
| 	return i.rdb.AllQueues() | ||||
| } | ||||
|  | ||||
| // QueueStats represents a state of queues at a certain time. | ||||
| type QueueStats struct { | ||||
| // QueueInfo represents a state of queues at a certain time. | ||||
| type QueueInfo struct { | ||||
| 	// Name of the queue. | ||||
| 	Queue string | ||||
|  | ||||
| 	// Total number of bytes that the queue and its tasks require to be stored in redis. | ||||
| 	// It is an approximate memory usage value in bytes since the value is computed by sampling. | ||||
| 	MemoryUsage int64 | ||||
|  | ||||
| 	// Size is the total number of tasks in the queue. | ||||
| 	// The value is the sum of Pending, Active, Scheduled, Retry, and Archived. | ||||
| 	Size int | ||||
|  | ||||
| 	// Number of pending tasks. | ||||
| 	Pending int | ||||
| 	// Number of active tasks. | ||||
| @@ -54,35 +66,42 @@ type QueueStats struct { | ||||
| 	Retry int | ||||
| 	// Number of archived tasks. | ||||
| 	Archived int | ||||
| 	// Number of stored completed tasks. | ||||
| 	Completed int | ||||
|  | ||||
| 	// Total number of tasks being processed during the given date. | ||||
| 	// The number includes both succeeded and failed tasks. | ||||
| 	Processed int | ||||
| 	// Total number of tasks failed to be processed during the given date. | ||||
| 	Failed int | ||||
|  | ||||
| 	// Paused indicates whether the queue is paused. | ||||
| 	// If true, tasks in the queue will not be processed. | ||||
| 	Paused bool | ||||
| 	// Time when this stats was taken. | ||||
|  | ||||
| 	// Time when this queue info snapshot was taken. | ||||
| 	Timestamp time.Time | ||||
| } | ||||
|  | ||||
| // CurrentStats returns a current stats of the given queue. | ||||
| func (i *Inspector) CurrentStats(qname string) (*QueueStats, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| // GetQueueInfo returns current information of the given queue. | ||||
| func (i *Inspector) GetQueueInfo(qname string) (*QueueInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	stats, err := i.rdb.CurrentStats(qname) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &QueueStats{ | ||||
| 	return &QueueInfo{ | ||||
| 		Queue:       stats.Queue, | ||||
| 		MemoryUsage: stats.MemoryUsage, | ||||
| 		Size:        stats.Size, | ||||
| 		Pending:     stats.Pending, | ||||
| 		Active:      stats.Active, | ||||
| 		Scheduled:   stats.Scheduled, | ||||
| 		Retry:       stats.Retry, | ||||
| 		Archived:    stats.Archived, | ||||
| 		Completed:   stats.Completed, | ||||
| 		Processed:   stats.Processed, | ||||
| 		Failed:      stats.Failed, | ||||
| 		Paused:      stats.Paused, | ||||
| @@ -105,7 +124,7 @@ type DailyStats struct { | ||||
|  | ||||
| // History returns a list of stats from the last n days. | ||||
| func (i *Inspector) History(qname string, n int) ([]*DailyStats, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	stats, err := i.rdb.HistoricalStats(qname, n) | ||||
| @@ -124,23 +143,16 @@ func (i *Inspector) History(qname string, n int) ([]*DailyStats, error) { | ||||
| 	return res, nil | ||||
| } | ||||
|  | ||||
| // ErrQueueNotFound indicates that the specified queue does not exist. | ||||
| type ErrQueueNotFound struct { | ||||
| 	qname string | ||||
| } | ||||
| var ( | ||||
| 	// ErrQueueNotFound indicates that the specified queue does not exist. | ||||
| 	ErrQueueNotFound = errors.New("queue not found") | ||||
|  | ||||
| func (e *ErrQueueNotFound) Error() string { | ||||
| 	return fmt.Sprintf("queue %q does not exist", e.qname) | ||||
| } | ||||
| 	// ErrQueueNotEmpty indicates that the specified queue is not empty. | ||||
| 	ErrQueueNotEmpty = errors.New("queue is not empty") | ||||
|  | ||||
| // ErrQueueNotEmpty indicates that the specified queue is not empty. | ||||
| type ErrQueueNotEmpty struct { | ||||
| 	qname string | ||||
| } | ||||
|  | ||||
| func (e *ErrQueueNotEmpty) Error() string { | ||||
| 	return fmt.Sprintf("queue %q is not empty", e.qname) | ||||
| } | ||||
| 	// ErrTaskNotFound indicates that the specified task cannot be found in the queue. | ||||
| 	ErrTaskNotFound = errors.New("task not found") | ||||
| ) | ||||
|  | ||||
| // DeleteQueue removes the specified queue. | ||||
| // | ||||
| @@ -154,104 +166,30 @@ func (e *ErrQueueNotEmpty) Error() string { | ||||
| // returns ErrQueueNotEmpty. | ||||
| func (i *Inspector) DeleteQueue(qname string, force bool) error { | ||||
| 	err := i.rdb.RemoveQueue(qname, force) | ||||
| 	if _, ok := err.(*rdb.ErrQueueNotFound); ok { | ||||
| 		return &ErrQueueNotFound{qname} | ||||
| 	if errors.IsQueueNotFound(err) { | ||||
| 		return fmt.Errorf("%w: queue=%q", ErrQueueNotFound, qname) | ||||
| 	} | ||||
| 	if _, ok := err.(*rdb.ErrQueueNotEmpty); ok { | ||||
| 		return &ErrQueueNotEmpty{qname} | ||||
| 	if errors.IsQueueNotEmpty(err) { | ||||
| 		return fmt.Errorf("%w: queue=%q", ErrQueueNotEmpty, qname) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // PendingTask is a task in a queue and is ready to be processed. | ||||
| type PendingTask struct { | ||||
| 	*Task | ||||
| 	ID    string | ||||
| 	Queue string | ||||
| } | ||||
|  | ||||
| // ActiveTask is a task that's currently being processed. | ||||
| type ActiveTask struct { | ||||
| 	*Task | ||||
| 	ID    string | ||||
| 	Queue string | ||||
| } | ||||
|  | ||||
| // ScheduledTask is a task scheduled to be processed in the future. | ||||
| type ScheduledTask struct { | ||||
| 	*Task | ||||
| 	ID            string | ||||
| 	Queue         string | ||||
| 	NextProcessAt time.Time | ||||
|  | ||||
| 	score int64 | ||||
| } | ||||
|  | ||||
| // RetryTask is a task scheduled to be retried in the future. | ||||
| type RetryTask struct { | ||||
| 	*Task | ||||
| 	ID            string | ||||
| 	Queue         string | ||||
| 	NextProcessAt time.Time | ||||
| 	MaxRetry      int | ||||
| 	Retried       int | ||||
| 	ErrorMsg      string | ||||
| 	// TODO: LastFailedAt  time.Time | ||||
|  | ||||
| 	score int64 | ||||
| } | ||||
|  | ||||
| // ArchivedTask is a task archived for debugging and inspection purposes, and | ||||
| // it won't be retried automatically. | ||||
| // A task can be archived when the task exhausts its retry counts or manually | ||||
| // archived by a user via the CLI or Inspector. | ||||
| type ArchivedTask struct { | ||||
| 	*Task | ||||
| 	ID           string | ||||
| 	Queue        string | ||||
| 	MaxRetry     int | ||||
| 	Retried      int | ||||
| 	LastFailedAt time.Time | ||||
| 	ErrorMsg     string | ||||
|  | ||||
| 	score int64 | ||||
| } | ||||
|  | ||||
| // Key returns a key used to delete, run, and archive the task. | ||||
| func (t *ScheduledTask) Key() string { | ||||
| 	return fmt.Sprintf("s:%v:%v", t.ID, t.score) | ||||
| } | ||||
|  | ||||
| // Key returns a key used to delete, run, and archive the task. | ||||
| func (t *RetryTask) Key() string { | ||||
| 	return fmt.Sprintf("r:%v:%v", t.ID, t.score) | ||||
| } | ||||
|  | ||||
| // Key returns a key used to delete, run, and archive the task. | ||||
| func (t *ArchivedTask) Key() string { | ||||
| 	return fmt.Sprintf("a:%v:%v", t.ID, t.score) | ||||
| } | ||||
|  | ||||
| // parseTaskKey parses a key string and returns each part of key with proper | ||||
| // type if valid, otherwise it reports an error. | ||||
| func parseTaskKey(key string) (id uuid.UUID, score int64, state string, err error) { | ||||
| 	parts := strings.Split(key, ":") | ||||
| 	if len(parts) != 3 { | ||||
| 		return uuid.Nil, 0, "", fmt.Errorf("invalid id") | ||||
| // GetTaskInfo retrieves task information given a task id and queue name. | ||||
| // | ||||
| // Returns ErrQueueNotFound if a queue with the given name doesn't exist. | ||||
| // Returns ErrTaskNotFound if a task with the given id doesn't exist in the queue. | ||||
| func (i *Inspector) GetTaskInfo(qname, id string) (*TaskInfo, error) { | ||||
| 	info, err := i.rdb.GetTaskInfo(qname, id) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case errors.IsTaskNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrTaskNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	id, err = uuid.Parse(parts[1]) | ||||
| 	if err != nil { | ||||
| 		return uuid.Nil, 0, "", fmt.Errorf("invalid id") | ||||
| 	} | ||||
| 	score, err = strconv.ParseInt(parts[2], 10, 64) | ||||
| 	if err != nil { | ||||
| 		return uuid.Nil, 0, "", fmt.Errorf("invalid id") | ||||
| 	} | ||||
| 	state = parts[0] | ||||
| 	if len(state) != 1 || !strings.Contains("sra", state) { | ||||
| 		return uuid.Nil, 0, "", fmt.Errorf("invalid id") | ||||
| 	} | ||||
| 	return id, score, state, nil | ||||
| 	return newTaskInfo(info.Message, info.State, info.NextProcessAt, info.Result), nil | ||||
| } | ||||
|  | ||||
| // ListOption specifies behavior of list operation. | ||||
| @@ -318,23 +256,27 @@ func Page(n int) ListOption { | ||||
| // ListPendingTasks retrieves pending tasks from the specified queue. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListPendingTasks(qname string, opts ...ListOption) ([]*PendingTask, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| func (i *Inspector) ListPendingTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	msgs, err := i.rdb.ListPending(qname, pgn) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	infos, err := i.rdb.ListPending(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*PendingTask | ||||
| 	for _, m := range msgs { | ||||
| 		tasks = append(tasks, &PendingTask{ | ||||
| 			Task:  NewTask(m.Type, m.Payload), | ||||
| 			ID:    m.ID.String(), | ||||
| 			Queue: m.Queue, | ||||
| 		}) | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, err | ||||
| } | ||||
| @@ -342,125 +284,161 @@ func (i *Inspector) ListPendingTasks(qname string, opts ...ListOption) ([]*Pendi | ||||
| // ListActiveTasks retrieves active tasks from the specified queue. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListActiveTasks(qname string, opts ...ListOption) ([]*ActiveTask, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| func (i *Inspector) ListActiveTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	msgs, err := i.rdb.ListActive(qname, pgn) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	infos, err := i.rdb.ListActive(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*ActiveTask | ||||
| 	for _, m := range msgs { | ||||
| 		tasks = append(tasks, &ActiveTask{ | ||||
| 			Task:  NewTask(m.Type, m.Payload), | ||||
| 			ID:    m.ID.String(), | ||||
| 			Queue: m.Queue, | ||||
| 		}) | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, err | ||||
| } | ||||
|  | ||||
| // ListScheduledTasks retrieves scheduled tasks from the specified queue. | ||||
| // Tasks are sorted by NextProcessAt field in ascending order. | ||||
| // Tasks are sorted by NextProcessAt in ascending order. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListScheduledTasks(qname string, opts ...ListOption) ([]*ScheduledTask, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| func (i *Inspector) ListScheduledTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	zs, err := i.rdb.ListScheduled(qname, pgn) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	infos, err := i.rdb.ListScheduled(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*ScheduledTask | ||||
| 	for _, z := range zs { | ||||
| 		processAt := time.Unix(z.Score, 0) | ||||
| 		t := NewTask(z.Message.Type, z.Message.Payload) | ||||
| 		tasks = append(tasks, &ScheduledTask{ | ||||
| 			Task:          t, | ||||
| 			ID:            z.Message.ID.String(), | ||||
| 			Queue:         z.Message.Queue, | ||||
| 			NextProcessAt: processAt, | ||||
| 			score:         z.Score, | ||||
| 		}) | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, nil | ||||
| } | ||||
|  | ||||
| // ListRetryTasks retrieves retry tasks from the specified queue. | ||||
| // Tasks are sorted by NextProcessAt field in ascending order. | ||||
| // Tasks are sorted by NextProcessAt in ascending order. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListRetryTasks(qname string, opts ...ListOption) ([]*RetryTask, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| func (i *Inspector) ListRetryTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	zs, err := i.rdb.ListRetry(qname, pgn) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	infos, err := i.rdb.ListRetry(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*RetryTask | ||||
| 	for _, z := range zs { | ||||
| 		processAt := time.Unix(z.Score, 0) | ||||
| 		t := NewTask(z.Message.Type, z.Message.Payload) | ||||
| 		tasks = append(tasks, &RetryTask{ | ||||
| 			Task:          t, | ||||
| 			ID:            z.Message.ID.String(), | ||||
| 			Queue:         z.Message.Queue, | ||||
| 			NextProcessAt: processAt, | ||||
| 			MaxRetry:      z.Message.Retry, | ||||
| 			Retried:       z.Message.Retried, | ||||
| 			// TODO: LastFailedAt: z.Message.LastFailedAt | ||||
| 			ErrorMsg: z.Message.ErrorMsg, | ||||
| 			score:    z.Score, | ||||
| 		}) | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, nil | ||||
| } | ||||
|  | ||||
| // ListArchivedTasks retrieves archived tasks from the specified queue. | ||||
| // Tasks are sorted by LastFailedAt field in descending order. | ||||
| // Tasks are sorted by LastFailedAt in descending order. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListArchivedTasks(qname string, opts ...ListOption) ([]*ArchivedTask, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return nil, err | ||||
| func (i *Inspector) ListArchivedTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	zs, err := i.rdb.ListArchived(qname, pgn) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	infos, err := i.rdb.ListArchived(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*ArchivedTask | ||||
| 	for _, z := range zs { | ||||
| 		failedAt := time.Unix(z.Score, 0) | ||||
| 		t := NewTask(z.Message.Type, z.Message.Payload) | ||||
| 		tasks = append(tasks, &ArchivedTask{ | ||||
| 			Task:         t, | ||||
| 			ID:           z.Message.ID.String(), | ||||
| 			Queue:        z.Message.Queue, | ||||
| 			MaxRetry:     z.Message.Retry, | ||||
| 			Retried:      z.Message.Retried, | ||||
| 			LastFailedAt: failedAt, | ||||
| 			ErrorMsg:     z.Message.ErrorMsg, | ||||
| 			score:        z.Score, | ||||
| 		}) | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, nil | ||||
| } | ||||
|  | ||||
| // ListCompletedTasks retrieves completed tasks from the specified queue. | ||||
| // Tasks are sorted by expiration time (i.e. CompletedAt + Retention) in descending order. | ||||
| // | ||||
| // By default, it retrieves the first 30 tasks. | ||||
| func (i *Inspector) ListCompletedTasks(qname string, opts ...ListOption) ([]*TaskInfo, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	opt := composeListOptions(opts...) | ||||
| 	pgn := rdb.Pagination{Size: opt.pageSize, Page: opt.pageNum - 1} | ||||
| 	infos, err := i.rdb.ListCompleted(qname, pgn) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return nil, fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case err != nil: | ||||
| 		return nil, fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	var tasks []*TaskInfo | ||||
| 	for _, i := range infos { | ||||
| 		tasks = append(tasks, newTaskInfo( | ||||
| 			i.Message, | ||||
| 			i.State, | ||||
| 			i.NextProcessAt, | ||||
| 			i.Result, | ||||
| 		)) | ||||
| 	} | ||||
| 	return tasks, nil | ||||
| } | ||||
|  | ||||
| // DeleteAllPendingTasks deletes all pending tasks from the specified queue, | ||||
| // and reports the number tasks deleted. | ||||
| func (i *Inspector) DeleteAllPendingTasks(qname string) (int, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.DeleteAllPendingTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // DeleteAllScheduledTasks deletes all scheduled tasks from the specified queue, | ||||
| // and reports the number tasks deleted. | ||||
| func (i *Inspector) DeleteAllScheduledTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.DeleteAllScheduledTasks(qname) | ||||
| @@ -470,7 +448,7 @@ func (i *Inspector) DeleteAllScheduledTasks(qname string) (int, error) { | ||||
| // DeleteAllRetryTasks deletes all retry tasks from the specified queue, | ||||
| // and reports the number tasks deleted. | ||||
| func (i *Inspector) DeleteAllRetryTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.DeleteAllRetryTasks(qname) | ||||
| @@ -480,138 +458,165 @@ func (i *Inspector) DeleteAllRetryTasks(qname string) (int, error) { | ||||
| // DeleteAllArchivedTasks deletes all archived tasks from the specified queue, | ||||
| // and reports the number tasks deleted. | ||||
| func (i *Inspector) DeleteAllArchivedTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.DeleteAllArchivedTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // DeleteTaskByKey deletes a task with the given key from the given queue. | ||||
| func (i *Inspector) DeleteTaskByKey(qname, key string) error { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	id, score, state, err := parseTaskKey(key) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	switch state { | ||||
| 	case "s": | ||||
| 		return i.rdb.DeleteScheduledTask(qname, id, score) | ||||
| 	case "r": | ||||
| 		return i.rdb.DeleteRetryTask(qname, id, score) | ||||
| 	case "a": | ||||
| 		return i.rdb.DeleteArchivedTask(qname, id, score) | ||||
| 	default: | ||||
| 		return fmt.Errorf("invalid key") | ||||
| // DeleteAllCompletedTasks deletes all completed tasks from the specified queue, | ||||
| // and reports the number tasks deleted. | ||||
| func (i *Inspector) DeleteAllCompletedTasks(qname string) (int, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.DeleteAllCompletedTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // RunAllScheduledTasks transition all scheduled tasks to pending state within the given queue, | ||||
| // DeleteTask deletes a task with the given id from the given queue. | ||||
| // The task needs to be in pending, scheduled, retry, or archived state, | ||||
| // otherwise DeleteTask will return an error. | ||||
| // | ||||
| // If a queue with the given name doesn't exist, it returns ErrQueueNotFound. | ||||
| // If a task with the given id doesn't exist in the queue, it returns ErrTaskNotFound. | ||||
| // If the task is in active state, it returns a non-nil error. | ||||
| func (i *Inspector) DeleteTask(qname, id string) error { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	err := i.rdb.DeleteTask(qname, id) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case errors.IsTaskNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrTaskNotFound) | ||||
| 	case err != nil: | ||||
| 		return fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	return nil | ||||
|  | ||||
| } | ||||
|  | ||||
| // RunAllScheduledTasks transition all scheduled tasks to pending state from the given queue, | ||||
| // and reports the number of tasks transitioned. | ||||
| func (i *Inspector) RunAllScheduledTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.RunAllScheduledTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // RunAllRetryTasks transition all retry tasks to pending state within the given queue, | ||||
| // RunAllRetryTasks transition all retry tasks to pending state from the given queue, | ||||
| // and reports the number of tasks transitioned. | ||||
| func (i *Inspector) RunAllRetryTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.RunAllRetryTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // RunAllArchivedTasks transition all archived tasks to pending state within the given queue, | ||||
| // RunAllArchivedTasks transition all archived tasks to pending state from the given queue, | ||||
| // and reports the number of tasks transitioned. | ||||
| func (i *Inspector) RunAllArchivedTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.RunAllArchivedTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // RunTaskByKey transition a task to pending state given task key and queue name. | ||||
| func (i *Inspector) RunTaskByKey(qname, key string) error { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return err | ||||
| // RunTask updates the task to pending state given a queue name and task id. | ||||
| // The task needs to be in scheduled, retry, or archived state, otherwise RunTask | ||||
| // will return an error. | ||||
| // | ||||
| // If a queue with the given name doesn't exist, it returns ErrQueueNotFound. | ||||
| // If a task with the given id doesn't exist in the queue, it returns ErrTaskNotFound. | ||||
| // If the task is in pending or active state, it returns a non-nil error. | ||||
| func (i *Inspector) RunTask(qname, id string) error { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	id, score, state, err := parseTaskKey(key) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	switch state { | ||||
| 	case "s": | ||||
| 		return i.rdb.RunScheduledTask(qname, id, score) | ||||
| 	case "r": | ||||
| 		return i.rdb.RunRetryTask(qname, id, score) | ||||
| 	case "a": | ||||
| 		return i.rdb.RunArchivedTask(qname, id, score) | ||||
| 	default: | ||||
| 		return fmt.Errorf("invalid key") | ||||
| 	err := i.rdb.RunTask(qname, id) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case errors.IsTaskNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrTaskNotFound) | ||||
| 	case err != nil: | ||||
| 		return fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // ArchiveAllScheduledTasks archives all scheduled tasks within the given queue, | ||||
| // ArchiveAllPendingTasks archives all pending tasks from the given queue, | ||||
| // and reports the number of tasks archived. | ||||
| func (i *Inspector) ArchiveAllPendingTasks(qname string) (int, error) { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.ArchiveAllPendingTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // ArchiveAllScheduledTasks archives all scheduled tasks from the given queue, | ||||
| // and reports the number of tasks archiveed. | ||||
| func (i *Inspector) ArchiveAllScheduledTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.ArchiveAllScheduledTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // ArchiveAllRetryTasks archives all retry tasks within the given queue, | ||||
| // ArchiveAllRetryTasks archives all retry tasks from the given queue, | ||||
| // and reports the number of tasks archiveed. | ||||
| func (i *Inspector) ArchiveAllRetryTasks(qname string) (int, error) { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	n, err := i.rdb.ArchiveAllRetryTasks(qname) | ||||
| 	return int(n), err | ||||
| } | ||||
|  | ||||
| // ArchiveTaskByKey archives a task with the given key in the given queue. | ||||
| func (i *Inspector) ArchiveTaskByKey(qname, key string) error { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 		return err | ||||
| // ArchiveTask archives a task with the given id in the given queue. | ||||
| // The task needs to be in pending, scheduled, or retry state, otherwise ArchiveTask | ||||
| // will return an error. | ||||
| // | ||||
| // If a queue with the given name doesn't exist, it returns ErrQueueNotFound. | ||||
| // If a task with the given id doesn't exist in the queue, it returns ErrTaskNotFound. | ||||
| // If the task is in already archived, it returns a non-nil error. | ||||
| func (i *Inspector) ArchiveTask(qname, id string) error { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return fmt.Errorf("asynq: err") | ||||
| 	} | ||||
| 	id, score, state, err := parseTaskKey(key) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	switch state { | ||||
| 	case "s": | ||||
| 		return i.rdb.ArchiveScheduledTask(qname, id, score) | ||||
| 	case "r": | ||||
| 		return i.rdb.ArchiveRetryTask(qname, id, score) | ||||
| 	case "a": | ||||
| 		return fmt.Errorf("task already archived") | ||||
| 	default: | ||||
| 		return fmt.Errorf("invalid key") | ||||
| 	err := i.rdb.ArchiveTask(qname, id) | ||||
| 	switch { | ||||
| 	case errors.IsQueueNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrQueueNotFound) | ||||
| 	case errors.IsTaskNotFound(err): | ||||
| 		return fmt.Errorf("asynq: %w", ErrTaskNotFound) | ||||
| 	case err != nil: | ||||
| 		return fmt.Errorf("asynq: %v", err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // CancelActiveTask sends a signal to cancel processing of the task with | ||||
| // the given id. CancelActiveTask is best-effort, which means that it does not | ||||
| // CancelProcessing sends a signal to cancel processing of the task | ||||
| // given a task id. CancelProcessing is best-effort, which means that it does not | ||||
| // guarantee that the task with the given id will be canceled. The return | ||||
| // value only indicates whether the cancelation signal has been sent. | ||||
| func (i *Inspector) CancelActiveTask(id string) error { | ||||
| func (i *Inspector) CancelProcessing(id string) error { | ||||
| 	return i.rdb.PublishCancelation(id) | ||||
| } | ||||
|  | ||||
| // PauseQueue pauses task processing on the specified queue. | ||||
| // If the queue is already paused, it will return a non-nil error. | ||||
| func (i *Inspector) PauseQueue(qname string) error { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return i.rdb.Pause(qname) | ||||
| @@ -620,7 +625,7 @@ func (i *Inspector) PauseQueue(qname string) error { | ||||
| // UnpauseQueue resumes task processing on the specified queue. | ||||
| // If the queue is not paused, it will return a non-nil error. | ||||
| func (i *Inspector) UnpauseQueue(qname string) error { | ||||
| 	if err := validateQueueName(qname); err != nil { | ||||
| 	if err := base.ValidateQueueName(qname); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return i.rdb.Unpause(qname) | ||||
| @@ -656,12 +661,12 @@ func (i *Inspector) Servers() ([]*ServerInfo, error) { | ||||
| 			continue | ||||
| 		} | ||||
| 		wrkInfo := &WorkerInfo{ | ||||
| 			Started: w.Started, | ||||
| 			Task: &ActiveTask{ | ||||
| 				Task:  NewTask(w.Type, w.Payload), | ||||
| 				ID:    w.ID, | ||||
| 			TaskID:      w.ID, | ||||
| 			TaskType:    w.Type, | ||||
| 			TaskPayload: w.Payload, | ||||
| 			Queue:       w.Queue, | ||||
| 			}, | ||||
| 			Started:     w.Started, | ||||
| 			Deadline:    w.Deadline, | ||||
| 		} | ||||
| 		srvInfo.ActiveWorkers = append(srvInfo.ActiveWorkers, wrkInfo) | ||||
| 	} | ||||
| @@ -698,10 +703,18 @@ type ServerInfo struct { | ||||
|  | ||||
| // WorkerInfo describes a running worker processing a task. | ||||
| type WorkerInfo struct { | ||||
| 	// The task the worker is processing. | ||||
| 	Task *ActiveTask | ||||
| 	// ID of the task the worker is processing. | ||||
| 	TaskID string | ||||
| 	// Type of the task the worker is processing. | ||||
| 	TaskType string | ||||
| 	// Payload of the task the worker is processing. | ||||
| 	TaskPayload []byte | ||||
| 	// Queue from which the worker got its task. | ||||
| 	Queue string | ||||
| 	// Time the worker started processing the task. | ||||
| 	Started time.Time | ||||
| 	// Time the worker needs to finish processing the task by. | ||||
| 	Deadline time.Time | ||||
| } | ||||
|  | ||||
| // ClusterKeySlot returns an integer identifying the hash slot the given queue hashes to. | ||||
| @@ -719,14 +732,16 @@ type ClusterNode struct { | ||||
| } | ||||
|  | ||||
| // ClusterNodes returns a list of nodes the given queue belongs to. | ||||
| func (i *Inspector) ClusterNodes(qname string) ([]ClusterNode, error) { | ||||
| // | ||||
| // Only relevant if task queues are stored in redis cluster. | ||||
| func (i *Inspector) ClusterNodes(qname string) ([]*ClusterNode, error) { | ||||
| 	nodes, err := i.rdb.ClusterNodes(qname) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	var res []ClusterNode | ||||
| 	var res []*ClusterNode | ||||
| 	for _, node := range nodes { | ||||
| 		res = append(res, ClusterNode{ID: node.ID, Addr: node.Addr}) | ||||
| 		res = append(res, &ClusterNode{ID: node.ID, Addr: node.Addr}) | ||||
| 	} | ||||
| 	return res, nil | ||||
| } | ||||
| @@ -782,6 +797,80 @@ func (i *Inspector) SchedulerEntries() ([]*SchedulerEntry, error) { | ||||
| 	return entries, nil | ||||
| } | ||||
|  | ||||
| // parseOption interprets a string s as an Option and returns the Option if parsing is successful, | ||||
| // otherwise returns non-nil error. | ||||
| func parseOption(s string) (Option, error) { | ||||
| 	fn, arg := parseOptionFunc(s), parseOptionArg(s) | ||||
| 	switch fn { | ||||
| 	case "Queue": | ||||
| 		qname, err := strconv.Unquote(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Queue(qname), nil | ||||
| 	case "MaxRetry": | ||||
| 		n, err := strconv.Atoi(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return MaxRetry(n), nil | ||||
| 	case "Timeout": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Timeout(d), nil | ||||
| 	case "Deadline": | ||||
| 		t, err := time.Parse(time.UnixDate, arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Deadline(t), nil | ||||
| 	case "Unique": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Unique(d), nil | ||||
| 	case "ProcessAt": | ||||
| 		t, err := time.Parse(time.UnixDate, arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return ProcessAt(t), nil | ||||
| 	case "ProcessIn": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return ProcessIn(d), nil | ||||
| 	case "Retention": | ||||
| 		d, err := time.ParseDuration(arg) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		return Retention(d), nil | ||||
| 	default: | ||||
| 		return nil, fmt.Errorf("cannot not parse option string %q", s) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func parseOptionFunc(s string) string { | ||||
| 	i := strings.Index(s, "(") | ||||
| 	return s[:i] | ||||
| } | ||||
|  | ||||
| func parseOptionArg(s string) string { | ||||
| 	i := strings.Index(s, "(") | ||||
| 	if i >= 0 { | ||||
| 		j := strings.Index(s, ")") | ||||
| 		if j > i { | ||||
| 			return s[i+1 : j] | ||||
| 		} | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| // SchedulerEnqueueEvent holds information about an enqueue event by a scheduler. | ||||
| type SchedulerEnqueueEvent struct { | ||||
| 	// ID of the task that was enqueued. | ||||
|   | ||||
							
								
								
									
										1718
									
								
								inspector_test.go
									
									
									
									
									
								
							
							
						
						
									
										1718
									
								
								inspector_test.go
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -6,12 +6,14 @@ | ||||
| package asynqtest | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"math" | ||||
| 	"sort" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	"github.com/google/go-cmp/cmp/cmpopts" | ||||
| 	"github.com/google/uuid" | ||||
| @@ -30,7 +32,7 @@ func EquateInt64Approx(margin int64) cmp.Option { | ||||
| var SortMsgOpt = cmp.Transformer("SortTaskMessages", func(in []*base.TaskMessage) []*base.TaskMessage { | ||||
| 	out := append([]*base.TaskMessage(nil), in...) // Copy input to avoid mutating it | ||||
| 	sort.Slice(out, func(i, j int) bool { | ||||
| 		return out[i].ID.String() < out[j].ID.String() | ||||
| 		return out[i].ID < out[j].ID | ||||
| 	}) | ||||
| 	return out | ||||
| }) | ||||
| @@ -39,7 +41,7 @@ var SortMsgOpt = cmp.Transformer("SortTaskMessages", func(in []*base.TaskMessage | ||||
| var SortZSetEntryOpt = cmp.Transformer("SortZSetEntries", func(in []base.Z) []base.Z { | ||||
| 	out := append([]base.Z(nil), in...) // Copy input to avoid mutating it | ||||
| 	sort.Slice(out, func(i, j int) bool { | ||||
| 		return out[i].Message.ID.String() < out[j].Message.ID.String() | ||||
| 		return out[i].Message.ID < out[j].Message.ID | ||||
| 	}) | ||||
| 	return out | ||||
| }) | ||||
| @@ -94,15 +96,15 @@ var SortStringSliceOpt = cmp.Transformer("SortStringSlice", func(in []string) [] | ||||
| var IgnoreIDOpt = cmpopts.IgnoreFields(base.TaskMessage{}, "ID") | ||||
|  | ||||
| // NewTaskMessage returns a new instance of TaskMessage given a task type and payload. | ||||
| func NewTaskMessage(taskType string, payload map[string]interface{}) *base.TaskMessage { | ||||
| func NewTaskMessage(taskType string, payload []byte) *base.TaskMessage { | ||||
| 	return NewTaskMessageWithQueue(taskType, payload, base.DefaultQueueName) | ||||
| } | ||||
|  | ||||
| // NewTaskMessageWithQueue returns a new instance of TaskMessage given a | ||||
| // task type, payload and queue name. | ||||
| func NewTaskMessageWithQueue(taskType string, payload map[string]interface{}, qname string) *base.TaskMessage { | ||||
| func NewTaskMessageWithQueue(taskType string, payload []byte, qname string) *base.TaskMessage { | ||||
| 	return &base.TaskMessage{ | ||||
| 		ID:       uuid.New(), | ||||
| 		ID:       uuid.NewString(), | ||||
| 		Type:     taskType, | ||||
| 		Queue:    qname, | ||||
| 		Retry:    25, | ||||
| @@ -112,17 +114,34 @@ func NewTaskMessageWithQueue(taskType string, payload map[string]interface{}, qn | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // JSON serializes the given key-value pairs into stream of bytes in JSON. | ||||
| func JSON(kv map[string]interface{}) []byte { | ||||
| 	b, err := json.Marshal(kv) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| // TaskMessageAfterRetry returns an updated copy of t after retry. | ||||
| // It increments retry count and sets the error message. | ||||
| func TaskMessageAfterRetry(t base.TaskMessage, errMsg string) *base.TaskMessage { | ||||
| // It increments retry count and sets the error message and last_failed_at time. | ||||
| func TaskMessageAfterRetry(t base.TaskMessage, errMsg string, failedAt time.Time) *base.TaskMessage { | ||||
| 	t.Retried = t.Retried + 1 | ||||
| 	t.ErrorMsg = errMsg | ||||
| 	t.LastFailedAt = failedAt.Unix() | ||||
| 	return &t | ||||
| } | ||||
|  | ||||
| // TaskMessageWithError returns an updated copy of t with the given error message. | ||||
| func TaskMessageWithError(t base.TaskMessage, errMsg string) *base.TaskMessage { | ||||
| func TaskMessageWithError(t base.TaskMessage, errMsg string, failedAt time.Time) *base.TaskMessage { | ||||
| 	t.ErrorMsg = errMsg | ||||
| 	t.LastFailedAt = failedAt.Unix() | ||||
| 	return &t | ||||
| } | ||||
|  | ||||
| // TaskMessageWithCompletedAt returns an updated copy of t after completion. | ||||
| func TaskMessageWithCompletedAt(t base.TaskMessage, completedAt time.Time) *base.TaskMessage { | ||||
| 	t.CompletedAt = completedAt.Unix() | ||||
| 	return &t | ||||
| } | ||||
|  | ||||
| @@ -130,7 +149,7 @@ func TaskMessageWithError(t base.TaskMessage, errMsg string) *base.TaskMessage { | ||||
| // Calling test will fail if marshaling errors out. | ||||
| func MustMarshal(tb testing.TB, msg *base.TaskMessage) string { | ||||
| 	tb.Helper() | ||||
| 	data, err := json.Marshal(msg) | ||||
| 	data, err := base.EncodeMessage(msg) | ||||
| 	if err != nil { | ||||
| 		tb.Fatal(err) | ||||
| 	} | ||||
| @@ -141,34 +160,11 @@ func MustMarshal(tb testing.TB, msg *base.TaskMessage) string { | ||||
| // Calling test will fail if unmarshaling errors out. | ||||
| func MustUnmarshal(tb testing.TB, data string) *base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	var msg base.TaskMessage | ||||
| 	err := json.Unmarshal([]byte(data), &msg) | ||||
| 	msg, err := base.DecodeMessage([]byte(data)) | ||||
| 	if err != nil { | ||||
| 		tb.Fatal(err) | ||||
| 	} | ||||
| 	return &msg | ||||
| } | ||||
|  | ||||
| // MustMarshalSlice marshals a slice of task messages and return a slice of | ||||
| // json strings. Calling test will fail if marshaling errors out. | ||||
| func MustMarshalSlice(tb testing.TB, msgs []*base.TaskMessage) []string { | ||||
| 	tb.Helper() | ||||
| 	var data []string | ||||
| 	for _, m := range msgs { | ||||
| 		data = append(data, MustMarshal(tb, m)) | ||||
| 	} | ||||
| 	return data | ||||
| } | ||||
|  | ||||
| // MustUnmarshalSlice unmarshals a slice of strings into a slice of task message structs. | ||||
| // Calling test will fail if marshaling errors out. | ||||
| func MustUnmarshalSlice(tb testing.TB, data []string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	var msgs []*base.TaskMessage | ||||
| 	for _, s := range data { | ||||
| 		msgs = append(msgs, MustUnmarshal(tb, s)) | ||||
| 	} | ||||
| 	return msgs | ||||
| 	return msg | ||||
| } | ||||
|  | ||||
| // FlushDB deletes all the keys of the currently selected DB. | ||||
| @@ -176,12 +172,12 @@ func FlushDB(tb testing.TB, r redis.UniversalClient) { | ||||
| 	tb.Helper() | ||||
| 	switch r := r.(type) { | ||||
| 	case *redis.Client: | ||||
| 		if err := r.FlushDB().Err(); err != nil { | ||||
| 		if err := r.FlushDB(context.Background()).Err(); err != nil { | ||||
| 			tb.Fatal(err) | ||||
| 		} | ||||
| 	case *redis.ClusterClient: | ||||
| 		err := r.ForEachMaster(func(c *redis.Client) error { | ||||
| 			if err := c.FlushAll().Err(); err != nil { | ||||
| 		err := r.ForEachMaster(context.Background(), func(ctx context.Context, c *redis.Client) error { | ||||
| 			if err := c.FlushAll(ctx).Err(); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			return nil | ||||
| @@ -195,49 +191,57 @@ func FlushDB(tb testing.TB, r redis.UniversalClient) { | ||||
| // SeedPendingQueue initializes the specified queue with the given messages. | ||||
| func SeedPendingQueue(tb testing.TB, r redis.UniversalClient, msgs []*base.TaskMessage, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisList(tb, r, base.QueueKey(qname), msgs) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisList(tb, r, base.PendingKey(qname), msgs, base.TaskStatePending) | ||||
| } | ||||
|  | ||||
| // SeedActiveQueue initializes the active queue with the given messages. | ||||
| func SeedActiveQueue(tb testing.TB, r redis.UniversalClient, msgs []*base.TaskMessage, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisList(tb, r, base.ActiveKey(qname), msgs) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisList(tb, r, base.ActiveKey(qname), msgs, base.TaskStateActive) | ||||
| } | ||||
|  | ||||
| // SeedScheduledQueue initializes the scheduled queue with the given messages. | ||||
| func SeedScheduledQueue(tb testing.TB, r redis.UniversalClient, entries []base.Z, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.ScheduledKey(qname), entries) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.ScheduledKey(qname), entries, base.TaskStateScheduled) | ||||
| } | ||||
|  | ||||
| // SeedRetryQueue initializes the retry queue with the given messages. | ||||
| func SeedRetryQueue(tb testing.TB, r redis.UniversalClient, entries []base.Z, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.RetryKey(qname), entries) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.RetryKey(qname), entries, base.TaskStateRetry) | ||||
| } | ||||
|  | ||||
| // SeedArchivedQueue initializes the archived queue with the given messages. | ||||
| func SeedArchivedQueue(tb testing.TB, r redis.UniversalClient, entries []base.Z, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.ArchivedKey(qname), entries) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.ArchivedKey(qname), entries, base.TaskStateArchived) | ||||
| } | ||||
|  | ||||
| // SeedDeadlines initializes the deadlines set with the given entries. | ||||
| func SeedDeadlines(tb testing.TB, r redis.UniversalClient, entries []base.Z, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.DeadlinesKey(qname), entries) | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.DeadlinesKey(qname), entries, base.TaskStateActive) | ||||
| } | ||||
|  | ||||
| // SeedCompletedQueue initializes the completed set witht the given entries. | ||||
| func SeedCompletedQueue(tb testing.TB, r redis.UniversalClient, entries []base.Z, qname string) { | ||||
| 	tb.Helper() | ||||
| 	r.SAdd(context.Background(), base.AllQueues, qname) | ||||
| 	seedRedisZSet(tb, r, base.CompletedKey(qname), entries, base.TaskStateCompleted) | ||||
| } | ||||
|  | ||||
| // SeedAllPendingQueues initializes all of the specified queues with the given messages. | ||||
| // | ||||
| // pending maps a queue name to a list of messages. | ||||
| func SeedAllPendingQueues(tb testing.TB, r redis.UniversalClient, pending map[string][]*base.TaskMessage) { | ||||
| 	tb.Helper() | ||||
| 	for q, msgs := range pending { | ||||
| 		SeedPendingQueue(tb, r, msgs, q) | ||||
| 	} | ||||
| @@ -245,6 +249,7 @@ func SeedAllPendingQueues(tb testing.TB, r redis.UniversalClient, pending map[st | ||||
|  | ||||
| // SeedAllActiveQueues initializes all of the specified active queues with the given messages. | ||||
| func SeedAllActiveQueues(tb testing.TB, r redis.UniversalClient, active map[string][]*base.TaskMessage) { | ||||
| 	tb.Helper() | ||||
| 	for q, msgs := range active { | ||||
| 		SeedActiveQueue(tb, r, msgs, q) | ||||
| 	} | ||||
| @@ -252,6 +257,7 @@ func SeedAllActiveQueues(tb testing.TB, r redis.UniversalClient, active map[stri | ||||
|  | ||||
| // SeedAllScheduledQueues initializes all of the specified scheduled queues with the given entries. | ||||
| func SeedAllScheduledQueues(tb testing.TB, r redis.UniversalClient, scheduled map[string][]base.Z) { | ||||
| 	tb.Helper() | ||||
| 	for q, entries := range scheduled { | ||||
| 		SeedScheduledQueue(tb, r, entries, q) | ||||
| 	} | ||||
| @@ -259,6 +265,7 @@ func SeedAllScheduledQueues(tb testing.TB, r redis.UniversalClient, scheduled ma | ||||
|  | ||||
| // SeedAllRetryQueues initializes all of the specified retry queues with the given entries. | ||||
| func SeedAllRetryQueues(tb testing.TB, r redis.UniversalClient, retry map[string][]base.Z) { | ||||
| 	tb.Helper() | ||||
| 	for q, entries := range retry { | ||||
| 		SeedRetryQueue(tb, r, entries, q) | ||||
| 	} | ||||
| @@ -266,6 +273,7 @@ func SeedAllRetryQueues(tb testing.TB, r redis.UniversalClient, retry map[string | ||||
|  | ||||
| // SeedAllArchivedQueues initializes all of the specified archived queues with the given entries. | ||||
| func SeedAllArchivedQueues(tb testing.TB, r redis.UniversalClient, archived map[string][]base.Z) { | ||||
| 	tb.Helper() | ||||
| 	for q, entries := range archived { | ||||
| 		SeedArchivedQueue(tb, r, entries, q) | ||||
| 	} | ||||
| @@ -273,101 +281,203 @@ func SeedAllArchivedQueues(tb testing.TB, r redis.UniversalClient, archived map[ | ||||
|  | ||||
| // SeedAllDeadlines initializes all of the deadlines with the given entries. | ||||
| func SeedAllDeadlines(tb testing.TB, r redis.UniversalClient, deadlines map[string][]base.Z) { | ||||
| 	tb.Helper() | ||||
| 	for q, entries := range deadlines { | ||||
| 		SeedDeadlines(tb, r, entries, q) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func seedRedisList(tb testing.TB, c redis.UniversalClient, key string, msgs []*base.TaskMessage) { | ||||
| 	data := MustMarshalSlice(tb, msgs) | ||||
| 	for _, s := range data { | ||||
| 		if err := c.LPush(key, s).Err(); err != nil { | ||||
| // SeedAllCompletedQueues initializes all of the completed queues with the given entries. | ||||
| func SeedAllCompletedQueues(tb testing.TB, r redis.UniversalClient, completed map[string][]base.Z) { | ||||
| 	tb.Helper() | ||||
| 	for q, entries := range completed { | ||||
| 		SeedCompletedQueue(tb, r, entries, q) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func seedRedisList(tb testing.TB, c redis.UniversalClient, key string, | ||||
| 	msgs []*base.TaskMessage, state base.TaskState) { | ||||
| 	tb.Helper() | ||||
| 	for _, msg := range msgs { | ||||
| 		encoded := MustMarshal(tb, msg) | ||||
| 		if err := c.LPush(context.Background(), key, msg.ID).Err(); err != nil { | ||||
| 			tb.Fatal(err) | ||||
| 		} | ||||
| 		key := base.TaskKey(msg.Queue, msg.ID) | ||||
| 		data := map[string]interface{}{ | ||||
| 			"msg":        encoded, | ||||
| 			"state":      state.String(), | ||||
| 			"timeout":    msg.Timeout, | ||||
| 			"deadline":   msg.Deadline, | ||||
| 			"unique_key": msg.UniqueKey, | ||||
| 		} | ||||
| 		if err := c.HSet(context.Background(), key, data).Err(); err != nil { | ||||
| 			tb.Fatal(err) | ||||
| 		} | ||||
| 		if len(msg.UniqueKey) > 0 { | ||||
| 			err := c.SetNX(context.Background(), msg.UniqueKey, msg.ID, 1*time.Minute).Err() | ||||
| 			if err != nil { | ||||
| 				tb.Fatalf("Failed to set unique lock in redis: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func seedRedisZSet(tb testing.TB, c redis.UniversalClient, key string, items []base.Z) { | ||||
| func seedRedisZSet(tb testing.TB, c redis.UniversalClient, key string, | ||||
| 	items []base.Z, state base.TaskState) { | ||||
| 	tb.Helper() | ||||
| 	for _, item := range items { | ||||
| 		z := &redis.Z{Member: MustMarshal(tb, item.Message), Score: float64(item.Score)} | ||||
| 		if err := c.ZAdd(key, z).Err(); err != nil { | ||||
| 		msg := item.Message | ||||
| 		encoded := MustMarshal(tb, msg) | ||||
| 		z := &redis.Z{Member: msg.ID, Score: float64(item.Score)} | ||||
| 		if err := c.ZAdd(context.Background(), key, z).Err(); err != nil { | ||||
| 			tb.Fatal(err) | ||||
| 		} | ||||
| 		key := base.TaskKey(msg.Queue, msg.ID) | ||||
| 		data := map[string]interface{}{ | ||||
| 			"msg":        encoded, | ||||
| 			"state":      state.String(), | ||||
| 			"timeout":    msg.Timeout, | ||||
| 			"deadline":   msg.Deadline, | ||||
| 			"unique_key": msg.UniqueKey, | ||||
| 		} | ||||
| 		if err := c.HSet(context.Background(), key, data).Err(); err != nil { | ||||
| 			tb.Fatal(err) | ||||
| 		} | ||||
| 		if len(msg.UniqueKey) > 0 { | ||||
| 			err := c.SetNX(context.Background(), msg.UniqueKey, msg.ID, 1*time.Minute).Err() | ||||
| 			if err != nil { | ||||
| 				tb.Fatalf("Failed to set unique lock in redis: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // GetPendingMessages returns all pending messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetPendingMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getListMessages(tb, r, base.QueueKey(qname)) | ||||
| 	return getMessagesFromList(tb, r, qname, base.PendingKey, base.TaskStatePending) | ||||
| } | ||||
|  | ||||
| // GetActiveMessages returns all active messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetActiveMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getListMessages(tb, r, base.ActiveKey(qname)) | ||||
| 	return getMessagesFromList(tb, r, qname, base.ActiveKey, base.TaskStateActive) | ||||
| } | ||||
|  | ||||
| // GetScheduledMessages returns all scheduled task messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetScheduledMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getZSetMessages(tb, r, base.ScheduledKey(qname)) | ||||
| 	return getMessagesFromZSet(tb, r, qname, base.ScheduledKey, base.TaskStateScheduled) | ||||
| } | ||||
|  | ||||
| // GetRetryMessages returns all retry messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetRetryMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getZSetMessages(tb, r, base.RetryKey(qname)) | ||||
| 	return getMessagesFromZSet(tb, r, qname, base.RetryKey, base.TaskStateRetry) | ||||
| } | ||||
|  | ||||
| // GetArchivedMessages returns all archived messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetArchivedMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getZSetMessages(tb, r, base.ArchivedKey(qname)) | ||||
| 	return getMessagesFromZSet(tb, r, qname, base.ArchivedKey, base.TaskStateArchived) | ||||
| } | ||||
|  | ||||
| // GetCompletedMessages returns all completed task messages in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetCompletedMessages(tb testing.TB, r redis.UniversalClient, qname string) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	return getMessagesFromZSet(tb, r, qname, base.CompletedKey, base.TaskStateCompleted) | ||||
| } | ||||
|  | ||||
| // GetScheduledEntries returns all scheduled messages and its score in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetScheduledEntries(tb testing.TB, r redis.UniversalClient, qname string) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	return getZSetEntries(tb, r, base.ScheduledKey(qname)) | ||||
| 	return getMessagesFromZSetWithScores(tb, r, qname, base.ScheduledKey, base.TaskStateScheduled) | ||||
| } | ||||
|  | ||||
| // GetRetryEntries returns all retry messages and its score in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetRetryEntries(tb testing.TB, r redis.UniversalClient, qname string) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	return getZSetEntries(tb, r, base.RetryKey(qname)) | ||||
| 	return getMessagesFromZSetWithScores(tb, r, qname, base.RetryKey, base.TaskStateRetry) | ||||
| } | ||||
|  | ||||
| // GetArchivedEntries returns all archived messages and its score in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetArchivedEntries(tb testing.TB, r redis.UniversalClient, qname string) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	return getZSetEntries(tb, r, base.ArchivedKey(qname)) | ||||
| 	return getMessagesFromZSetWithScores(tb, r, qname, base.ArchivedKey, base.TaskStateArchived) | ||||
| } | ||||
|  | ||||
| // GetDeadlinesEntries returns all task messages and its score in the deadlines set for the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetDeadlinesEntries(tb testing.TB, r redis.UniversalClient, qname string) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	return getZSetEntries(tb, r, base.DeadlinesKey(qname)) | ||||
| 	return getMessagesFromZSetWithScores(tb, r, qname, base.DeadlinesKey, base.TaskStateActive) | ||||
| } | ||||
|  | ||||
| func getListMessages(tb testing.TB, r redis.UniversalClient, list string) []*base.TaskMessage { | ||||
| 	data := r.LRange(list, 0, -1).Val() | ||||
| 	return MustUnmarshalSlice(tb, data) | ||||
| // GetCompletedEntries returns all completed messages and its score in the given queue. | ||||
| // It also asserts the state field of the task. | ||||
| func GetCompletedEntries(tb testing.TB, r redis.UniversalClient, qname string) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	return getMessagesFromZSetWithScores(tb, r, qname, base.CompletedKey, base.TaskStateCompleted) | ||||
| } | ||||
|  | ||||
| func getZSetMessages(tb testing.TB, r redis.UniversalClient, zset string) []*base.TaskMessage { | ||||
| 	data := r.ZRange(zset, 0, -1).Val() | ||||
| 	return MustUnmarshalSlice(tb, data) | ||||
| } | ||||
|  | ||||
| func getZSetEntries(tb testing.TB, r redis.UniversalClient, zset string) []base.Z { | ||||
| 	data := r.ZRangeWithScores(zset, 0, -1).Val() | ||||
| 	var entries []base.Z | ||||
| 	for _, z := range data { | ||||
| 		entries = append(entries, base.Z{ | ||||
| 			Message: MustUnmarshal(tb, z.Member.(string)), | ||||
| 			Score:   int64(z.Score), | ||||
| 		}) | ||||
| // Retrieves all messages stored under `keyFn(qname)` key in redis list. | ||||
| func getMessagesFromList(tb testing.TB, r redis.UniversalClient, qname string, | ||||
| 	keyFn func(qname string) string, state base.TaskState) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	ids := r.LRange(context.Background(), keyFn(qname), 0, -1).Val() | ||||
| 	var msgs []*base.TaskMessage | ||||
| 	for _, id := range ids { | ||||
| 		taskKey := base.TaskKey(qname, id) | ||||
| 		data := r.HGet(context.Background(), taskKey, "msg").Val() | ||||
| 		msgs = append(msgs, MustUnmarshal(tb, data)) | ||||
| 		if gotState := r.HGet(context.Background(), taskKey, "state").Val(); gotState != state.String() { | ||||
| 			tb.Errorf("task (id=%q) is in %q state, want %v", id, gotState, state) | ||||
| 		} | ||||
| 	return entries | ||||
| 	} | ||||
| 	return msgs | ||||
| } | ||||
|  | ||||
| // Retrieves all messages stored under `keyFn(qname)` key in redis zset (sorted-set). | ||||
| func getMessagesFromZSet(tb testing.TB, r redis.UniversalClient, qname string, | ||||
| 	keyFn func(qname string) string, state base.TaskState) []*base.TaskMessage { | ||||
| 	tb.Helper() | ||||
| 	ids := r.ZRange(context.Background(), keyFn(qname), 0, -1).Val() | ||||
| 	var msgs []*base.TaskMessage | ||||
| 	for _, id := range ids { | ||||
| 		taskKey := base.TaskKey(qname, id) | ||||
| 		msg := r.HGet(context.Background(), taskKey, "msg").Val() | ||||
| 		msgs = append(msgs, MustUnmarshal(tb, msg)) | ||||
| 		if gotState := r.HGet(context.Background(), taskKey, "state").Val(); gotState != state.String() { | ||||
| 			tb.Errorf("task (id=%q) is in %q state, want %v", id, gotState, state) | ||||
| 		} | ||||
| 	} | ||||
| 	return msgs | ||||
| } | ||||
|  | ||||
| // Retrieves all messages along with their scores stored under `keyFn(qname)` key in redis zset (sorted-set). | ||||
| func getMessagesFromZSetWithScores(tb testing.TB, r redis.UniversalClient, | ||||
| 	qname string, keyFn func(qname string) string, state base.TaskState) []base.Z { | ||||
| 	tb.Helper() | ||||
| 	zs := r.ZRangeWithScores(context.Background(), keyFn(qname), 0, -1).Val() | ||||
| 	var res []base.Z | ||||
| 	for _, z := range zs { | ||||
| 		taskID := z.Member.(string) | ||||
| 		taskKey := base.TaskKey(qname, taskID) | ||||
| 		msg := r.HGet(context.Background(), taskKey, "msg").Val() | ||||
| 		res = append(res, base.Z{Message: MustUnmarshal(tb, msg), Score: int64(z.Score)}) | ||||
| 		if gotState := r.HGet(context.Background(), taskKey, "state").Val(); gotState != state.String() { | ||||
| 			tb.Errorf("task (id=%q) is in %q state, want %v", taskID, gotState, state) | ||||
| 		} | ||||
| 	} | ||||
| 	return res | ||||
| } | ||||
|   | ||||
| @@ -7,25 +7,28 @@ package base | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"crypto/md5" | ||||
| 	"encoding/hex" | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/golang/protobuf/ptypes" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	pb "github.com/hibiken/asynq/internal/proto" | ||||
| 	"google.golang.org/protobuf/proto" | ||||
| ) | ||||
|  | ||||
| // Version of asynq library and CLI. | ||||
| const Version = "0.14.1" | ||||
| const Version = "0.19.0" | ||||
|  | ||||
| // DefaultQueueName is the queue name used if none are specified by user. | ||||
| const DefaultQueueName = "default" | ||||
|  | ||||
| // DefaultQueue is the redis key for the default queue. | ||||
| var DefaultQueue = QueueKey(DefaultQueueName) | ||||
| var DefaultQueue = PendingKey(DefaultQueueName) | ||||
|  | ||||
| // Global Redis keys. | ||||
| const ( | ||||
| @@ -36,49 +39,125 @@ const ( | ||||
| 	CancelChannel = "asynq:cancel"     // PubSub channel | ||||
| ) | ||||
|  | ||||
| // QueueKey returns a redis key for the given queue name. | ||||
| func QueueKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}", qname) | ||||
| // TaskState denotes the state of a task. | ||||
| type TaskState int | ||||
|  | ||||
| const ( | ||||
| 	TaskStateActive TaskState = iota + 1 | ||||
| 	TaskStatePending | ||||
| 	TaskStateScheduled | ||||
| 	TaskStateRetry | ||||
| 	TaskStateArchived | ||||
| 	TaskStateCompleted | ||||
| ) | ||||
|  | ||||
| func (s TaskState) String() string { | ||||
| 	switch s { | ||||
| 	case TaskStateActive: | ||||
| 		return "active" | ||||
| 	case TaskStatePending: | ||||
| 		return "pending" | ||||
| 	case TaskStateScheduled: | ||||
| 		return "scheduled" | ||||
| 	case TaskStateRetry: | ||||
| 		return "retry" | ||||
| 	case TaskStateArchived: | ||||
| 		return "archived" | ||||
| 	case TaskStateCompleted: | ||||
| 		return "completed" | ||||
| 	} | ||||
| 	panic(fmt.Sprintf("internal error: unknown task state %d", s)) | ||||
| } | ||||
|  | ||||
| func TaskStateFromString(s string) (TaskState, error) { | ||||
| 	switch s { | ||||
| 	case "active": | ||||
| 		return TaskStateActive, nil | ||||
| 	case "pending": | ||||
| 		return TaskStatePending, nil | ||||
| 	case "scheduled": | ||||
| 		return TaskStateScheduled, nil | ||||
| 	case "retry": | ||||
| 		return TaskStateRetry, nil | ||||
| 	case "archived": | ||||
| 		return TaskStateArchived, nil | ||||
| 	case "completed": | ||||
| 		return TaskStateCompleted, nil | ||||
| 	} | ||||
| 	return 0, errors.E(errors.FailedPrecondition, fmt.Sprintf("%q is not supported task state", s)) | ||||
| } | ||||
|  | ||||
| // ValidateQueueName validates a given qname to be used as a queue name. | ||||
| // Returns nil if valid, otherwise returns non-nil error. | ||||
| func ValidateQueueName(qname string) error { | ||||
| 	if len(strings.TrimSpace(qname)) == 0 { | ||||
| 		return fmt.Errorf("queue name must contain one or more characters") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // QueueKeyPrefix returns a prefix for all keys in the given queue. | ||||
| func QueueKeyPrefix(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:", qname) | ||||
| } | ||||
|  | ||||
| // TaskKeyPrefix returns a prefix for task key. | ||||
| func TaskKeyPrefix(qname string) string { | ||||
| 	return fmt.Sprintf("%st:", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // TaskKey returns a redis key for the given task message. | ||||
| func TaskKey(qname, id string) string { | ||||
| 	return fmt.Sprintf("%s%s", TaskKeyPrefix(qname), id) | ||||
| } | ||||
|  | ||||
| // PendingKey returns a redis key for the given queue name. | ||||
| func PendingKey(qname string) string { | ||||
| 	return fmt.Sprintf("%spending", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // ActiveKey returns a redis key for the active tasks. | ||||
| func ActiveKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:active", qname) | ||||
| 	return fmt.Sprintf("%sactive", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // ScheduledKey returns a redis key for the scheduled tasks. | ||||
| func ScheduledKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:scheduled", qname) | ||||
| 	return fmt.Sprintf("%sscheduled", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // RetryKey returns a redis key for the retry tasks. | ||||
| func RetryKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:retry", qname) | ||||
| 	return fmt.Sprintf("%sretry", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // ArchivedKey returns a redis key for the archived tasks. | ||||
| func ArchivedKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:archived", qname) | ||||
| 	return fmt.Sprintf("%sarchived", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // DeadlinesKey returns a redis key for the deadlines. | ||||
| func DeadlinesKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:deadlines", qname) | ||||
| 	return fmt.Sprintf("%sdeadlines", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| func CompletedKey(qname string) string { | ||||
| 	return fmt.Sprintf("%scompleted", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // PausedKey returns a redis key to indicate that the given queue is paused. | ||||
| func PausedKey(qname string) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:paused", qname) | ||||
| 	return fmt.Sprintf("%spaused", QueueKeyPrefix(qname)) | ||||
| } | ||||
|  | ||||
| // 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("asynq:{%s}:processed:%s", qname, t.UTC().Format("2006-01-02")) | ||||
| 	return fmt.Sprintf("%sprocessed:%s", QueueKeyPrefix(qname), 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("asynq:{%s}:failed:%s", qname, t.UTC().Format("2006-01-02")) | ||||
| 	return fmt.Sprintf("%sfailed:%s", QueueKeyPrefix(qname), t.UTC().Format("2006-01-02")) | ||||
| } | ||||
|  | ||||
| // ServerInfoKey returns a redis key for process info. | ||||
| @@ -102,32 +181,12 @@ func SchedulerHistoryKey(entryID string) string { | ||||
| } | ||||
|  | ||||
| // UniqueKey returns a redis key with the given type, payload, and queue name. | ||||
| func UniqueKey(qname, tasktype string, payload map[string]interface{}) string { | ||||
| 	return fmt.Sprintf("asynq:{%s}:unique:%s:%s", qname, tasktype, serializePayload(payload)) | ||||
| } | ||||
|  | ||||
| func serializePayload(payload map[string]interface{}) string { | ||||
| func UniqueKey(qname, tasktype string, payload []byte) string { | ||||
| 	if payload == nil { | ||||
| 		return "nil" | ||||
| 		return fmt.Sprintf("%sunique:%s:", QueueKeyPrefix(qname), tasktype) | ||||
| 	} | ||||
| 	type entry struct { | ||||
| 		k string | ||||
| 		v interface{} | ||||
| 	} | ||||
| 	var es []entry | ||||
| 	for k, v := range payload { | ||||
| 		es = append(es, entry{k, v}) | ||||
| 	} | ||||
| 	// sort entries by key | ||||
| 	sort.Slice(es, func(i, j int) bool { return es[i].k < es[j].k }) | ||||
| 	var b strings.Builder | ||||
| 	for _, e := range es { | ||||
| 		if b.Len() > 0 { | ||||
| 			b.WriteString(",") | ||||
| 		} | ||||
| 		b.WriteString(fmt.Sprintf("%s=%v", e.k, e.v)) | ||||
| 	} | ||||
| 	return b.String() | ||||
| 	checksum := md5.Sum(payload) | ||||
| 	return fmt.Sprintf("%sunique:%s:%s", QueueKeyPrefix(qname), tasktype, hex.EncodeToString(checksum[:])) | ||||
| } | ||||
|  | ||||
| // TaskMessage is the internal representation of a task with additional metadata fields. | ||||
| @@ -137,10 +196,10 @@ type TaskMessage struct { | ||||
| 	Type string | ||||
|  | ||||
| 	// Payload holds data needed to process the task. | ||||
| 	Payload map[string]interface{} | ||||
| 	Payload []byte | ||||
|  | ||||
| 	// ID is a unique identifier for each task. | ||||
| 	ID uuid.UUID | ||||
| 	ID string | ||||
|  | ||||
| 	// Queue is a name this message should be enqueued to. | ||||
| 	Queue string | ||||
| @@ -154,6 +213,12 @@ type TaskMessage struct { | ||||
| 	// ErrorMsg holds the error message from the last failure. | ||||
| 	ErrorMsg string | ||||
|  | ||||
| 	// Time of last failure in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// | ||||
| 	// Use zero to indicate no last failure | ||||
| 	LastFailedAt int64 | ||||
|  | ||||
| 	// Timeout specifies timeout in seconds. | ||||
| 	// If task processing doesn't complete within the timeout, the task will be retried | ||||
| 	// if retry count is remaining. Otherwise it will be moved to the archive. | ||||
| @@ -173,26 +238,68 @@ type TaskMessage struct { | ||||
| 	// | ||||
| 	// Empty string indicates that no uniqueness lock was used. | ||||
| 	UniqueKey string | ||||
|  | ||||
| 	// Retention specifies the number of seconds the task should be retained after completion. | ||||
| 	Retention int64 | ||||
|  | ||||
| 	// CompletedAt is the time the task was processed successfully in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// | ||||
| 	// Use zero to indicate no value. | ||||
| 	CompletedAt int64 | ||||
| } | ||||
|  | ||||
| // EncodeMessage marshals the given task message in JSON and returns an encoded string. | ||||
| func EncodeMessage(msg *TaskMessage) (string, error) { | ||||
| 	b, err := json.Marshal(msg) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| // EncodeMessage marshals the given task message and returns an encoded bytes. | ||||
| func EncodeMessage(msg *TaskMessage) ([]byte, error) { | ||||
| 	if msg == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil message") | ||||
| 	} | ||||
| 	return string(b), nil | ||||
| 	return proto.Marshal(&pb.TaskMessage{ | ||||
| 		Type:         msg.Type, | ||||
| 		Payload:      msg.Payload, | ||||
| 		Id:           msg.ID, | ||||
| 		Queue:        msg.Queue, | ||||
| 		Retry:        int32(msg.Retry), | ||||
| 		Retried:      int32(msg.Retried), | ||||
| 		ErrorMsg:     msg.ErrorMsg, | ||||
| 		LastFailedAt: msg.LastFailedAt, | ||||
| 		Timeout:      msg.Timeout, | ||||
| 		Deadline:     msg.Deadline, | ||||
| 		UniqueKey:    msg.UniqueKey, | ||||
| 		Retention:    msg.Retention, | ||||
| 		CompletedAt:  msg.CompletedAt, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // DecodeMessage unmarshals the given encoded string and returns a decoded task message. | ||||
| func DecodeMessage(s string) (*TaskMessage, error) { | ||||
| 	d := json.NewDecoder(strings.NewReader(s)) | ||||
| 	d.UseNumber() | ||||
| 	var msg TaskMessage | ||||
| 	if err := d.Decode(&msg); err != nil { | ||||
| // DecodeMessage unmarshals the given bytes and returns a decoded task message. | ||||
| func DecodeMessage(data []byte) (*TaskMessage, error) { | ||||
| 	var pbmsg pb.TaskMessage | ||||
| 	if err := proto.Unmarshal(data, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &msg, nil | ||||
| 	return &TaskMessage{ | ||||
| 		Type:         pbmsg.GetType(), | ||||
| 		Payload:      pbmsg.GetPayload(), | ||||
| 		ID:           pbmsg.GetId(), | ||||
| 		Queue:        pbmsg.GetQueue(), | ||||
| 		Retry:        int(pbmsg.GetRetry()), | ||||
| 		Retried:      int(pbmsg.GetRetried()), | ||||
| 		ErrorMsg:     pbmsg.GetErrorMsg(), | ||||
| 		LastFailedAt: pbmsg.GetLastFailedAt(), | ||||
| 		Timeout:      pbmsg.GetTimeout(), | ||||
| 		Deadline:     pbmsg.GetDeadline(), | ||||
| 		UniqueKey:    pbmsg.GetUniqueKey(), | ||||
| 		Retention:    pbmsg.GetRetention(), | ||||
| 		CompletedAt:  pbmsg.GetCompletedAt(), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // TaskInfo describes a task message and its metadata. | ||||
| type TaskInfo struct { | ||||
| 	Message       *TaskMessage | ||||
| 	State         TaskState | ||||
| 	NextProcessAt time.Time | ||||
| 	Result        []byte | ||||
| } | ||||
|  | ||||
| // Z represents sorted set member. | ||||
| @@ -201,52 +308,55 @@ type Z struct { | ||||
| 	Score   int64 | ||||
| } | ||||
|  | ||||
| // ServerStatus represents status of a server. | ||||
| // ServerStatus methods are concurrency safe. | ||||
| type ServerStatus struct { | ||||
| // ServerState represents state of a server. | ||||
| // ServerState methods are concurrency safe. | ||||
| type ServerState struct { | ||||
| 	mu  sync.Mutex | ||||
| 	val ServerStatusValue | ||||
| 	val ServerStateValue | ||||
| } | ||||
|  | ||||
| // NewServerStatus returns a new status instance given an initial value. | ||||
| func NewServerStatus(v ServerStatusValue) *ServerStatus { | ||||
| 	return &ServerStatus{val: v} | ||||
| // NewServerState returns a new state instance. | ||||
| // Initial state is set to StateNew. | ||||
| func NewServerState() *ServerState { | ||||
| 	return &ServerState{val: StateNew} | ||||
| } | ||||
|  | ||||
| type ServerStatusValue int | ||||
| type ServerStateValue int | ||||
|  | ||||
| const ( | ||||
| 	// StatusIdle indicates the server is in idle state. | ||||
| 	StatusIdle ServerStatusValue = iota | ||||
| 	// StateNew represents a new server. Server begins in | ||||
| 	// this state and then transition to StatusActive when | ||||
| 	// Start or Run is callled. | ||||
| 	StateNew ServerStateValue = iota | ||||
|  | ||||
| 	// StatusRunning indicates the server is up and active. | ||||
| 	StatusRunning | ||||
| 	// StateActive indicates the server is up and active. | ||||
| 	StateActive | ||||
|  | ||||
| 	// StatusQuiet indicates the server is up but not active. | ||||
| 	StatusQuiet | ||||
| 	// StateStopped indicates the server is up but no longer processing new tasks. | ||||
| 	StateStopped | ||||
|  | ||||
| 	// StatusStopped indicates the server server has been stopped. | ||||
| 	StatusStopped | ||||
| 	// StateClosed indicates the server has been shutdown. | ||||
| 	StateClosed | ||||
| ) | ||||
|  | ||||
| var statuses = []string{ | ||||
| 	"idle", | ||||
| 	"running", | ||||
| 	"quiet", | ||||
| var serverStates = []string{ | ||||
| 	"new", | ||||
| 	"active", | ||||
| 	"stopped", | ||||
| 	"closed", | ||||
| } | ||||
|  | ||||
| func (s *ServerStatus) String() string { | ||||
| func (s *ServerState) String() string { | ||||
| 	s.mu.Lock() | ||||
| 	defer s.mu.Unlock() | ||||
| 	if StatusIdle <= s.val && s.val <= StatusStopped { | ||||
| 		return statuses[s.val] | ||||
| 	if StateNew <= s.val && s.val <= StateClosed { | ||||
| 		return serverStates[s.val] | ||||
| 	} | ||||
| 	return "unknown status" | ||||
| } | ||||
|  | ||||
| // Get returns the status value. | ||||
| func (s *ServerStatus) Get() ServerStatusValue { | ||||
| func (s *ServerState) Get() ServerStateValue { | ||||
| 	s.mu.Lock() | ||||
| 	v := s.val | ||||
| 	s.mu.Unlock() | ||||
| @@ -254,7 +364,7 @@ func (s *ServerStatus) Get() ServerStatusValue { | ||||
| } | ||||
|  | ||||
| // Set sets the status value. | ||||
| func (s *ServerStatus) Set(v ServerStatusValue) { | ||||
| func (s *ServerState) Set(v ServerStateValue) { | ||||
| 	s.mu.Lock() | ||||
| 	s.val = v | ||||
| 	s.mu.Unlock() | ||||
| @@ -273,6 +383,59 @@ type ServerInfo struct { | ||||
| 	ActiveWorkerCount int | ||||
| } | ||||
|  | ||||
| // EncodeServerInfo marshals the given ServerInfo and returns the encoded bytes. | ||||
| func EncodeServerInfo(info *ServerInfo) ([]byte, error) { | ||||
| 	if info == nil { | ||||
| 		return nil, fmt.Errorf("cannot encode nil server info") | ||||
| 	} | ||||
| 	queues := make(map[string]int32) | ||||
| 	for q, p := range info.Queues { | ||||
| 		queues[q] = int32(p) | ||||
| 	} | ||||
| 	started, err := ptypes.TimestampProto(info.Started) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return proto.Marshal(&pb.ServerInfo{ | ||||
| 		Host:              info.Host, | ||||
| 		Pid:               int32(info.PID), | ||||
| 		ServerId:          info.ServerID, | ||||
| 		Concurrency:       int32(info.Concurrency), | ||||
| 		Queues:            queues, | ||||
| 		StrictPriority:    info.StrictPriority, | ||||
| 		Status:            info.Status, | ||||
| 		StartTime:         started, | ||||
| 		ActiveWorkerCount: int32(info.ActiveWorkerCount), | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // DecodeServerInfo decodes the given bytes into ServerInfo. | ||||
| func DecodeServerInfo(b []byte) (*ServerInfo, error) { | ||||
| 	var pbmsg pb.ServerInfo | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	queues := make(map[string]int) | ||||
| 	for q, p := range pbmsg.GetQueues() { | ||||
| 		queues[q] = int(p) | ||||
| 	} | ||||
| 	startTime, err := ptypes.Timestamp(pbmsg.GetStartTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &ServerInfo{ | ||||
| 		Host:              pbmsg.GetHost(), | ||||
| 		PID:               int(pbmsg.GetPid()), | ||||
| 		ServerID:          pbmsg.GetServerId(), | ||||
| 		Concurrency:       int(pbmsg.GetConcurrency()), | ||||
| 		Queues:            queues, | ||||
| 		StrictPriority:    pbmsg.GetStrictPriority(), | ||||
| 		Status:            pbmsg.GetStatus(), | ||||
| 		Started:           startTime, | ||||
| 		ActiveWorkerCount: int(pbmsg.GetActiveWorkerCount()), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // WorkerInfo holds information about a running worker. | ||||
| type WorkerInfo struct { | ||||
| 	Host     string | ||||
| @@ -280,9 +443,63 @@ type WorkerInfo struct { | ||||
| 	ServerID string | ||||
| 	ID       string | ||||
| 	Type     string | ||||
| 	Payload  []byte | ||||
| 	Queue    string | ||||
| 	Payload  map[string]interface{} | ||||
| 	Started  time.Time | ||||
| 	Deadline time.Time | ||||
| } | ||||
|  | ||||
| // EncodeWorkerInfo marshals the given WorkerInfo and returns the encoded bytes. | ||||
| 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 | ||||
| 	} | ||||
| 	return proto.Marshal(&pb.WorkerInfo{ | ||||
| 		Host:        info.Host, | ||||
| 		Pid:         int32(info.PID), | ||||
| 		ServerId:    info.ServerID, | ||||
| 		TaskId:      info.ID, | ||||
| 		TaskType:    info.Type, | ||||
| 		TaskPayload: info.Payload, | ||||
| 		Queue:       info.Queue, | ||||
| 		StartTime:   startTime, | ||||
| 		Deadline:    deadline, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // DecodeWorkerInfo decodes the given bytes into WorkerInfo. | ||||
| func DecodeWorkerInfo(b []byte) (*WorkerInfo, error) { | ||||
| 	var pbmsg pb.WorkerInfo | ||||
| 	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 | ||||
| 	} | ||||
| 	return &WorkerInfo{ | ||||
| 		Host:     pbmsg.GetHost(), | ||||
| 		PID:      int(pbmsg.GetPid()), | ||||
| 		ServerID: pbmsg.GetServerId(), | ||||
| 		ID:       pbmsg.GetTaskId(), | ||||
| 		Type:     pbmsg.GetTaskType(), | ||||
| 		Payload:  pbmsg.GetTaskPayload(), | ||||
| 		Queue:    pbmsg.GetQueue(), | ||||
| 		Started:  startTime, | ||||
| 		Deadline: deadline, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // SchedulerEntry holds information about a periodic task registered with a scheduler. | ||||
| @@ -297,7 +514,7 @@ type SchedulerEntry struct { | ||||
| 	Type string | ||||
|  | ||||
| 	// Payload is the payload of the periodic task. | ||||
| 	Payload map[string]interface{} | ||||
| 	Payload []byte | ||||
|  | ||||
| 	// Opts is the options for the periodic task. | ||||
| 	Opts []string | ||||
| @@ -310,6 +527,55 @@ type SchedulerEntry struct { | ||||
| 	Prev time.Time | ||||
| } | ||||
|  | ||||
| // EncodeSchedulerEntry marshals the given entry and returns an encoded bytes. | ||||
| 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 | ||||
| 	} | ||||
| 	return proto.Marshal(&pb.SchedulerEntry{ | ||||
| 		Id:              entry.ID, | ||||
| 		Spec:            entry.Spec, | ||||
| 		TaskType:        entry.Type, | ||||
| 		TaskPayload:     entry.Payload, | ||||
| 		EnqueueOptions:  entry.Opts, | ||||
| 		NextEnqueueTime: next, | ||||
| 		PrevEnqueueTime: prev, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // DecodeSchedulerEntry unmarshals the given bytes and returns a decoded SchedulerEntry. | ||||
| func DecodeSchedulerEntry(b []byte) (*SchedulerEntry, error) { | ||||
| 	var pbmsg pb.SchedulerEntry | ||||
| 	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 | ||||
| 	} | ||||
| 	return &SchedulerEntry{ | ||||
| 		ID:      pbmsg.GetId(), | ||||
| 		Spec:    pbmsg.GetSpec(), | ||||
| 		Type:    pbmsg.GetTaskType(), | ||||
| 		Payload: pbmsg.GetTaskPayload(), | ||||
| 		Opts:    pbmsg.GetEnqueueOptions(), | ||||
| 		Next:    next, | ||||
| 		Prev:    prev, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // SchedulerEnqueueEvent holds information about an enqueue event by a scheduler. | ||||
| type SchedulerEnqueueEvent struct { | ||||
| 	// ID of the task that was enqueued. | ||||
| @@ -319,6 +585,39 @@ type SchedulerEnqueueEvent struct { | ||||
| 	EnqueuedAt time.Time | ||||
| } | ||||
|  | ||||
| // EncodeSchedulerEnqueueEvent marshals the given event | ||||
| // and returns an encoded bytes. | ||||
| 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 | ||||
| 	} | ||||
| 	return proto.Marshal(&pb.SchedulerEnqueueEvent{ | ||||
| 		TaskId:      event.TaskID, | ||||
| 		EnqueueTime: enqueuedAt, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // DecodeSchedulerEnqueueEvent unmarshals the given bytes | ||||
| // and returns a decoded SchedulerEnqueueEvent. | ||||
| func DecodeSchedulerEnqueueEvent(b []byte) (*SchedulerEnqueueEvent, error) { | ||||
| 	var pbmsg pb.SchedulerEnqueueEvent | ||||
| 	if err := proto.Unmarshal(b, &pbmsg); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	enqueuedAt, err := ptypes.Timestamp(pbmsg.GetEnqueueTime()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &SchedulerEnqueueEvent{ | ||||
| 		TaskID:     pbmsg.GetTaskId(), | ||||
| 		EnqueuedAt: enqueuedAt, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Cancelations is a collection that holds cancel functions for all active tasks. | ||||
| // | ||||
| // Cancelations are safe for concurrent use by multipel goroutines. | ||||
| @@ -365,16 +664,19 @@ type Broker interface { | ||||
| 	EnqueueUnique(msg *TaskMessage, ttl time.Duration) error | ||||
| 	Dequeue(qnames ...string) (*TaskMessage, time.Time, error) | ||||
| 	Done(msg *TaskMessage) error | ||||
| 	MarkAsComplete(msg *TaskMessage) error | ||||
| 	Requeue(msg *TaskMessage) error | ||||
| 	Schedule(msg *TaskMessage, processAt time.Time) error | ||||
| 	ScheduleUnique(msg *TaskMessage, processAt time.Time, ttl time.Duration) error | ||||
| 	Retry(msg *TaskMessage, processAt time.Time, errMsg string) error | ||||
| 	Retry(msg *TaskMessage, processAt time.Time, errMsg string, isFailure bool) error | ||||
| 	Archive(msg *TaskMessage, errMsg string) error | ||||
| 	CheckAndEnqueue(qnames ...string) error | ||||
| 	ForwardIfReady(qnames ...string) error | ||||
| 	DeleteExpiredCompletedTasks(qname string) error | ||||
| 	ListDeadlineExceeded(deadline time.Time, qnames ...string) ([]*TaskMessage, error) | ||||
| 	WriteServerState(info *ServerInfo, workers []*WorkerInfo, ttl time.Duration) error | ||||
| 	ClearServerState(host string, pid int, serverID string) error | ||||
| 	CancelationPubSub() (*redis.PubSub, error) // TODO: Need to decouple from redis to support other brokers | ||||
| 	PublishCancelation(id string) error | ||||
| 	WriteResult(qname, id string, data []byte) (n int, err error) | ||||
| 	Close() error | ||||
| } | ||||
|   | ||||
| @@ -6,7 +6,10 @@ package base | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"crypto/md5" | ||||
| 	"encoding/hex" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| @@ -15,17 +18,36 @@ import ( | ||||
| 	"github.com/google/uuid" | ||||
| ) | ||||
|  | ||||
| func TestTaskKey(t *testing.T) { | ||||
| 	id := uuid.NewString() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		qname string | ||||
| 		id    string | ||||
| 		want  string | ||||
| 	}{ | ||||
| 		{"default", id, fmt.Sprintf("asynq:{default}:t:%s", id)}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		got := TaskKey(tc.qname, tc.id) | ||||
| 		if got != tc.want { | ||||
| 			t.Errorf("TaskKey(%q, %s) = %q, want %q", tc.qname, tc.id, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestQueueKey(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		qname string | ||||
| 		want  string | ||||
| 	}{ | ||||
| 		{"default", "asynq:{default}"}, | ||||
| 		{"custom", "asynq:{custom}"}, | ||||
| 		{"default", "asynq:{default}:pending"}, | ||||
| 		{"custom", "asynq:{custom}:pending"}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		got := QueueKey(tc.qname) | ||||
| 		got := PendingKey(tc.qname) | ||||
| 		if got != tc.want { | ||||
| 			t.Errorf("QueueKey(%q) = %q, want %q", tc.qname, got, tc.want) | ||||
| 		} | ||||
| @@ -117,6 +139,23 @@ func TestArchivedKey(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCompletedKey(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		qname string | ||||
| 		want  string | ||||
| 	}{ | ||||
| 		{"default", "asynq:{default}:completed"}, | ||||
| 		{"custom", "asynq:{custom}:completed"}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		got := CompletedKey(tc.qname) | ||||
| 		if got != tc.want { | ||||
| 			t.Errorf("CompletedKey(%q) = %q, want %q", tc.qname, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPausedKey(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		qname string | ||||
| @@ -247,52 +286,69 @@ func TestSchedulerHistoryKey(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func toBytes(m map[string]interface{}) []byte { | ||||
| 	b, err := json.Marshal(m) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	return b | ||||
| } | ||||
|  | ||||
| func TestUniqueKey(t *testing.T) { | ||||
| 	payload1 := toBytes(map[string]interface{}{"a": 123, "b": "hello", "c": true}) | ||||
| 	payload2 := toBytes(map[string]interface{}{"b": "hello", "c": true, "a": 123}) | ||||
| 	payload3 := toBytes(map[string]interface{}{ | ||||
| 		"address": map[string]string{"line": "123 Main St", "city": "Boston", "state": "MA"}, | ||||
| 		"names":   []string{"bob", "mike", "rob"}}) | ||||
| 	payload4 := toBytes(map[string]interface{}{ | ||||
| 		"time":     time.Date(2020, time.July, 28, 0, 0, 0, 0, time.UTC), | ||||
| 		"duration": time.Hour}) | ||||
|  | ||||
| 	checksum := func(data []byte) string { | ||||
| 		sum := md5.Sum(data) | ||||
| 		return hex.EncodeToString(sum[:]) | ||||
| 	} | ||||
| 	tests := []struct { | ||||
| 		desc     string | ||||
| 		qname    string | ||||
| 		tasktype string | ||||
| 		payload  map[string]interface{} | ||||
| 		payload  []byte | ||||
| 		want     string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"with primitive types", | ||||
| 			"default", | ||||
| 			"email:send", | ||||
| 			map[string]interface{}{"a": 123, "b": "hello", "c": true}, | ||||
| 			"asynq:{default}:unique:email:send:a=123,b=hello,c=true", | ||||
| 			payload1, | ||||
| 			fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload1)), | ||||
| 		}, | ||||
| 		{ | ||||
| 			"with unsorted keys", | ||||
| 			"default", | ||||
| 			"email:send", | ||||
| 			map[string]interface{}{"b": "hello", "c": true, "a": 123}, | ||||
| 			"asynq:{default}:unique:email:send:a=123,b=hello,c=true", | ||||
| 			payload2, | ||||
| 			fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload2)), | ||||
| 		}, | ||||
| 		{ | ||||
| 			"with composite types", | ||||
| 			"default", | ||||
| 			"email:send", | ||||
| 			map[string]interface{}{ | ||||
| 				"address": map[string]string{"line": "123 Main St", "city": "Boston", "state": "MA"}, | ||||
| 				"names":   []string{"bob", "mike", "rob"}}, | ||||
| 			"asynq:{default}:unique:email:send:address=map[city:Boston line:123 Main St state:MA],names=[bob mike rob]", | ||||
| 			payload3, | ||||
| 			fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload3)), | ||||
| 		}, | ||||
| 		{ | ||||
| 			"with complex types", | ||||
| 			"default", | ||||
| 			"email:send", | ||||
| 			map[string]interface{}{ | ||||
| 				"time":     time.Date(2020, time.July, 28, 0, 0, 0, 0, time.UTC), | ||||
| 				"duration": time.Hour}, | ||||
| 			"asynq:{default}:unique:email:send:duration=1h0m0s,time=2020-07-28 00:00:00 +0000 UTC", | ||||
| 			payload4, | ||||
| 			fmt.Sprintf("asynq:{default}:unique:email:send:%s", checksum(payload4)), | ||||
| 		}, | ||||
| 		{ | ||||
| 			"with nil payload", | ||||
| 			"default", | ||||
| 			"reindex", | ||||
| 			nil, | ||||
| 			"asynq:{default}:unique:reindex:nil", | ||||
| 			"asynq:{default}:unique:reindex:", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| @@ -305,7 +361,7 @@ func TestUniqueKey(t *testing.T) { | ||||
| } | ||||
|  | ||||
| func TestMessageEncoding(t *testing.T) { | ||||
| 	id := uuid.New() | ||||
| 	id := uuid.NewString() | ||||
| 	tests := []struct { | ||||
| 		in  *TaskMessage | ||||
| 		out *TaskMessage | ||||
| @@ -313,23 +369,25 @@ func TestMessageEncoding(t *testing.T) { | ||||
| 		{ | ||||
| 			in: &TaskMessage{ | ||||
| 				Type:      "task1", | ||||
| 				Payload:  map[string]interface{}{"a": 1, "b": "hello!", "c": true}, | ||||
| 				Payload:   toBytes(map[string]interface{}{"a": 1, "b": "hello!", "c": true}), | ||||
| 				ID:        id, | ||||
| 				Queue:     "default", | ||||
| 				Retry:     10, | ||||
| 				Retried:   0, | ||||
| 				Timeout:   1800, | ||||
| 				Deadline:  1692311100, | ||||
| 				Retention: 3600, | ||||
| 			}, | ||||
| 			out: &TaskMessage{ | ||||
| 				Type:      "task1", | ||||
| 				Payload:  map[string]interface{}{"a": json.Number("1"), "b": "hello!", "c": true}, | ||||
| 				Payload:   toBytes(map[string]interface{}{"a": json.Number("1"), "b": "hello!", "c": true}), | ||||
| 				ID:        id, | ||||
| 				Queue:     "default", | ||||
| 				Retry:     10, | ||||
| 				Retried:   0, | ||||
| 				Timeout:   1800, | ||||
| 				Deadline:  1692311100, | ||||
| 				Retention: 3600, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| @@ -352,10 +410,149 @@ func TestMessageEncoding(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestServerInfoEncoding(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		info ServerInfo | ||||
| 	}{ | ||||
| 		{ | ||||
| 			info: ServerInfo{ | ||||
| 				Host:              "127.0.0.1", | ||||
| 				PID:               9876, | ||||
| 				ServerID:          "abc123", | ||||
| 				Concurrency:       10, | ||||
| 				Queues:            map[string]int{"default": 1, "critical": 2}, | ||||
| 				StrictPriority:    false, | ||||
| 				Status:            "active", | ||||
| 				Started:           time.Now().Add(-3 * time.Hour), | ||||
| 				ActiveWorkerCount: 8, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		encoded, err := EncodeServerInfo(&tc.info) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("EncodeServerInfo(info) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		decoded, err := DecodeServerInfo(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("DecodeServerInfo(encoded) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if diff := cmp.Diff(&tc.info, decoded); diff != "" { | ||||
| 			t.Errorf("Decoded ServerInfo == %+v, want %+v;(-want,+got)\n%s", | ||||
| 				decoded, tc.info, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestWorkerInfoEncoding(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		info WorkerInfo | ||||
| 	}{ | ||||
| 		{ | ||||
| 			info: WorkerInfo{ | ||||
| 				Host:     "127.0.0.1", | ||||
| 				PID:      9876, | ||||
| 				ServerID: "abc123", | ||||
| 				ID:       uuid.NewString(), | ||||
| 				Type:     "taskA", | ||||
| 				Payload:  toBytes(map[string]interface{}{"foo": "bar"}), | ||||
| 				Queue:    "default", | ||||
| 				Started:  time.Now().Add(-3 * time.Hour), | ||||
| 				Deadline: time.Now().Add(30 * time.Second), | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		encoded, err := EncodeWorkerInfo(&tc.info) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("EncodeWorkerInfo(info) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		decoded, err := DecodeWorkerInfo(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("DecodeWorkerInfo(encoded) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if diff := cmp.Diff(&tc.info, decoded); diff != "" { | ||||
| 			t.Errorf("Decoded WorkerInfo == %+v, want %+v;(-want,+got)\n%s", | ||||
| 				decoded, tc.info, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestSchedulerEntryEncoding(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		entry SchedulerEntry | ||||
| 	}{ | ||||
| 		{ | ||||
| 			entry: SchedulerEntry{ | ||||
| 				ID:      uuid.NewString(), | ||||
| 				Spec:    "* * * * *", | ||||
| 				Type:    "task_A", | ||||
| 				Payload: toBytes(map[string]interface{}{"foo": "bar"}), | ||||
| 				Opts:    []string{"Queue('email')"}, | ||||
| 				Next:    time.Now().Add(30 * time.Second).UTC(), | ||||
| 				Prev:    time.Now().Add(-2 * time.Minute).UTC(), | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		encoded, err := EncodeSchedulerEntry(&tc.entry) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("EncodeSchedulerEntry(entry) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		decoded, err := DecodeSchedulerEntry(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("DecodeSchedulerEntry(encoded) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if diff := cmp.Diff(&tc.entry, decoded); diff != "" { | ||||
| 			t.Errorf("Decoded SchedulerEntry == %+v, want %+v;(-want,+got)\n%s", | ||||
| 				decoded, tc.entry, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestSchedulerEnqueueEventEncoding(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		event SchedulerEnqueueEvent | ||||
| 	}{ | ||||
| 		{ | ||||
| 			event: SchedulerEnqueueEvent{ | ||||
| 				TaskID:     uuid.NewString(), | ||||
| 				EnqueuedAt: time.Now().Add(-30 * time.Second).UTC(), | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		encoded, err := EncodeSchedulerEnqueueEvent(&tc.event) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("EncodeSchedulerEnqueueEvent(event) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		decoded, err := DecodeSchedulerEnqueueEvent(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("DecodeSchedulerEnqueueEvent(encoded) returned error: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if diff := cmp.Diff(&tc.event, decoded); diff != "" { | ||||
| 			t.Errorf("Decoded SchedulerEnqueueEvent == %+v, want %+v;(-want,+got)\n%s", | ||||
| 				decoded, tc.event, diff) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Test for status being accessed by multiple goroutines. | ||||
| // Run with -race flag to check for data race. | ||||
| func TestStatusConcurrentAccess(t *testing.T) { | ||||
| 	status := NewServerStatus(StatusIdle) | ||||
| 	status := NewServerState() | ||||
|  | ||||
| 	var wg sync.WaitGroup | ||||
|  | ||||
| @@ -369,7 +566,7 @@ func TestStatusConcurrentAccess(t *testing.T) { | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		status.Set(StatusStopped) | ||||
| 		status.Set(StateClosed) | ||||
| 		_ = status.String() | ||||
| 	}() | ||||
|  | ||||
|   | ||||
							
								
								
									
										87
									
								
								internal/context/context.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								internal/context/context.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package context | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| // A taskMetadata holds task scoped data to put in context. | ||||
| type taskMetadata struct { | ||||
| 	id         string | ||||
| 	maxRetry   int | ||||
| 	retryCount int | ||||
| 	qname      string | ||||
| } | ||||
|  | ||||
| // ctxKey type is unexported to prevent collisions with context keys defined in | ||||
| // other packages. | ||||
| type ctxKey int | ||||
|  | ||||
| // metadataCtxKey is the context key for the task metadata. | ||||
| // Its value of zero is arbitrary. | ||||
| const metadataCtxKey ctxKey = 0 | ||||
|  | ||||
| // New returns a context and cancel function for a given task message. | ||||
| func New(msg *base.TaskMessage, deadline time.Time) (context.Context, context.CancelFunc) { | ||||
| 	metadata := taskMetadata{ | ||||
| 		id:         msg.ID, | ||||
| 		maxRetry:   msg.Retry, | ||||
| 		retryCount: msg.Retried, | ||||
| 		qname:      msg.Queue, | ||||
| 	} | ||||
| 	ctx := context.WithValue(context.Background(), metadataCtxKey, metadata) | ||||
| 	return context.WithDeadline(ctx, deadline) | ||||
| } | ||||
|  | ||||
| // GetTaskID extracts a task ID from a context, if any. | ||||
| // | ||||
| // ID of a task is guaranteed to be unique. | ||||
| // ID of a task doesn't change if the task is being retried. | ||||
| func GetTaskID(ctx context.Context) (id string, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return "", false | ||||
| 	} | ||||
| 	return metadata.id, true | ||||
| } | ||||
|  | ||||
| // GetRetryCount extracts retry count from a context, if any. | ||||
| // | ||||
| // Return value n indicates the number of times associated task has been | ||||
| // retried so far. | ||||
| func GetRetryCount(ctx context.Context) (n int, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return 0, false | ||||
| 	} | ||||
| 	return metadata.retryCount, true | ||||
| } | ||||
|  | ||||
| // GetMaxRetry extracts maximum retry from a context, if any. | ||||
| // | ||||
| // Return value n indicates the maximum number of times the assoicated task | ||||
| // can be retried if ProcessTask returns a non-nil error. | ||||
| func GetMaxRetry(ctx context.Context) (n int, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return 0, false | ||||
| 	} | ||||
| 	return metadata.maxRetry, true | ||||
| } | ||||
|  | ||||
| // GetQueueName extracts queue name from a context, if any. | ||||
| // | ||||
| // Return value qname indicates which queue the task was pulled from. | ||||
| func GetQueueName(ctx context.Context) (qname string, ok bool) { | ||||
| 	metadata, ok := ctx.Value(metadataCtxKey).(taskMetadata) | ||||
| 	if !ok { | ||||
| 		return "", false | ||||
| 	} | ||||
| 	return metadata.qname, true | ||||
| } | ||||
| @@ -2,7 +2,7 @@ | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
| 
 | ||||
| package asynq | ||||
| package context | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| @@ -24,12 +24,11 @@ func TestCreateContextWithFutureDeadline(t *testing.T) { | ||||
| 	for _, tc := range tests { | ||||
| 		msg := &base.TaskMessage{ | ||||
| 			Type:    "something", | ||||
| 			ID:      uuid.New(), | ||||
| 			ID:      uuid.NewString(), | ||||
| 			Payload: nil, | ||||
| 		} | ||||
| 
 | ||||
| 		ctx, cancel := createContext(msg, tc.deadline) | ||||
| 
 | ||||
| 		ctx, cancel := New(msg, tc.deadline) | ||||
| 		select { | ||||
| 		case x := <-ctx.Done(): | ||||
| 			t.Errorf("<-ctx.Done() == %v, want nothing (it should block)", x) | ||||
| @@ -64,11 +63,11 @@ func TestCreateContextWithPastDeadline(t *testing.T) { | ||||
| 	for _, tc := range tests { | ||||
| 		msg := &base.TaskMessage{ | ||||
| 			Type:    "something", | ||||
| 			ID:      uuid.New(), | ||||
| 			ID:      uuid.NewString(), | ||||
| 			Payload: nil, | ||||
| 		} | ||||
| 
 | ||||
| 		ctx, cancel := createContext(msg, tc.deadline) | ||||
| 		ctx, cancel := New(msg, tc.deadline) | ||||
| 		defer cancel() | ||||
| 
 | ||||
| 		select { | ||||
| @@ -92,21 +91,21 @@ func TestGetTaskMetadataFromContext(t *testing.T) { | ||||
| 		desc string | ||||
| 		msg  *base.TaskMessage | ||||
| 	}{ | ||||
| 		{"with zero retried message", &base.TaskMessage{Type: "something", ID: uuid.New(), Retry: 25, Retried: 0, Timeout: 1800, Queue: "default"}}, | ||||
| 		{"with non-zero retried message", &base.TaskMessage{Type: "something", ID: uuid.New(), Retry: 10, Retried: 5, Timeout: 1800, Queue: "default"}}, | ||||
| 		{"with custom queue name", &base.TaskMessage{Type: "something", ID: uuid.New(), Retry: 25, Retried: 0, Timeout: 1800, Queue: "custom"}}, | ||||
| 		{"with zero retried message", &base.TaskMessage{Type: "something", ID: uuid.NewString(), Retry: 25, Retried: 0, Timeout: 1800, Queue: "default"}}, | ||||
| 		{"with non-zero retried message", &base.TaskMessage{Type: "something", ID: uuid.NewString(), Retry: 10, Retried: 5, Timeout: 1800, Queue: "default"}}, | ||||
| 		{"with custom queue name", &base.TaskMessage{Type: "something", ID: uuid.NewString(), Retry: 25, Retried: 0, Timeout: 1800, Queue: "custom"}}, | ||||
| 	} | ||||
| 
 | ||||
| 	for _, tc := range tests { | ||||
| 		ctx, cancel := createContext(tc.msg, time.Now().Add(30*time.Minute)) | ||||
| 		ctx, cancel := New(tc.msg, time.Now().Add(30*time.Minute)) | ||||
| 		defer cancel() | ||||
| 
 | ||||
| 		id, ok := GetTaskID(ctx) | ||||
| 		if !ok { | ||||
| 			t.Errorf("%s: GetTaskID(ctx) returned ok == false", tc.desc) | ||||
| 		} | ||||
| 		if ok && id != tc.msg.ID.String() { | ||||
| 			t.Errorf("%s: GetTaskID(ctx) returned id == %q, want %q", tc.desc, id, tc.msg.ID.String()) | ||||
| 		if ok && id != tc.msg.ID { | ||||
| 			t.Errorf("%s: GetTaskID(ctx) returned id == %q, want %q", tc.desc, id, tc.msg.ID) | ||||
| 		} | ||||
| 
 | ||||
| 		retried, ok := GetRetryCount(ctx) | ||||
							
								
								
									
										288
									
								
								internal/errors/errors.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										288
									
								
								internal/errors/errors.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,288 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| // Package errors defines the error type and functions used by | ||||
| // asynq and its internal packages. | ||||
| package errors | ||||
|  | ||||
| // Note: This package is inspired by a blog post about error handling in project Upspin | ||||
| // https://commandcenter.blogspot.com/2017/12/error-handling-in-upspin.html. | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // Error is the type that implements the error interface. | ||||
| // It contains a number of fields, each of different type. | ||||
| // An Error value may leave some values unset. | ||||
| type Error struct { | ||||
| 	Code Code | ||||
| 	Op   Op | ||||
| 	Err  error | ||||
| } | ||||
|  | ||||
| func (e *Error) DebugString() string { | ||||
| 	var b strings.Builder | ||||
| 	if e.Op != "" { | ||||
| 		b.WriteString(string(e.Op)) | ||||
| 	} | ||||
| 	if e.Code != Unspecified { | ||||
| 		if b.Len() > 0 { | ||||
| 			b.WriteString(": ") | ||||
| 		} | ||||
| 		b.WriteString(e.Code.String()) | ||||
| 	} | ||||
| 	if e.Err != nil { | ||||
| 		if b.Len() > 0 { | ||||
| 			b.WriteString(": ") | ||||
| 		} | ||||
| 		b.WriteString(e.Err.Error()) | ||||
| 	} | ||||
| 	return b.String() | ||||
| } | ||||
|  | ||||
| func (e *Error) Error() string { | ||||
| 	var b strings.Builder | ||||
| 	if e.Code != Unspecified { | ||||
| 		b.WriteString(e.Code.String()) | ||||
| 	} | ||||
| 	if e.Err != nil { | ||||
| 		if b.Len() > 0 { | ||||
| 			b.WriteString(": ") | ||||
| 		} | ||||
| 		b.WriteString(e.Err.Error()) | ||||
| 	} | ||||
| 	return b.String() | ||||
| } | ||||
|  | ||||
| func (e *Error) Unwrap() error { | ||||
| 	return e.Err | ||||
| } | ||||
|  | ||||
| // Code defines the canonical error code. | ||||
| type Code uint8 | ||||
|  | ||||
| // List of canonical error codes. | ||||
| const ( | ||||
| 	Unspecified Code = iota | ||||
| 	NotFound | ||||
| 	FailedPrecondition | ||||
| 	Internal | ||||
| 	AlreadyExists | ||||
| 	Unknown | ||||
| 	// Note: If you add a new value here, make sure to update String method. | ||||
| ) | ||||
|  | ||||
| func (c Code) String() string { | ||||
| 	switch c { | ||||
| 	case Unspecified: | ||||
| 		return "ERROR_CODE_UNSPECIFIED" | ||||
| 	case NotFound: | ||||
| 		return "NOT_FOUND" | ||||
| 	case FailedPrecondition: | ||||
| 		return "FAILED_PRECONDITION" | ||||
| 	case Internal: | ||||
| 		return "INTERNAL_ERROR" | ||||
| 	case AlreadyExists: | ||||
| 		return "ALREADY_EXISTS" | ||||
| 	case Unknown: | ||||
| 		return "UNKNOWN" | ||||
| 	} | ||||
| 	panic(fmt.Sprintf("unknown error code %d", c)) | ||||
| } | ||||
|  | ||||
| // Op describes an operation, usually as the package and method, | ||||
| // such as "rdb.Enqueue". | ||||
| type Op string | ||||
|  | ||||
| // E builds an error value from its arguments. | ||||
| // There must be at least one argument or E panics. | ||||
| // The type of each argument determines its meaning. | ||||
| // If more than one argument of a given type is presented, | ||||
| // only the last one is recorded. | ||||
| // | ||||
| // The types are: | ||||
| //	errors.Op | ||||
| //		The operation being performed, usually the method | ||||
| //		being invoked (Get, Put, etc.). | ||||
| //	errors.Code | ||||
| //		The canonical error code, such as NOT_FOUND. | ||||
| //	string | ||||
| //		Treated as an error message and assigned to the | ||||
| //		Err field after a call to errors.New. | ||||
| //	error | ||||
| //		The underlying error that triggered this one. | ||||
| // | ||||
| // If the error is printed, only those items that have been | ||||
| // set to non-zero values will appear in the result. | ||||
| func E(args ...interface{}) error { | ||||
| 	if len(args) == 0 { | ||||
| 		panic("call to errors.E with no arguments") | ||||
| 	} | ||||
| 	e := &Error{} | ||||
| 	for _, arg := range args { | ||||
| 		switch arg := arg.(type) { | ||||
| 		case Op: | ||||
| 			e.Op = arg | ||||
| 		case Code: | ||||
| 			e.Code = arg | ||||
| 		case error: | ||||
| 			e.Err = arg | ||||
| 		case string: | ||||
| 			e.Err = errors.New(arg) | ||||
| 		default: | ||||
| 			_, file, line, _ := runtime.Caller(1) | ||||
| 			log.Printf("errors.E: bad call from %s:%d: %v", file, line, args) | ||||
| 			return fmt.Errorf("unknown type %T, value %v in error call", arg, arg) | ||||
| 		} | ||||
| 	} | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| // CanonicalCode returns the canonical code of the given error if one is present. | ||||
| // Otherwise it returns Unspecified. | ||||
| func CanonicalCode(err error) Code { | ||||
| 	if err == nil { | ||||
| 		return Unspecified | ||||
| 	} | ||||
| 	e, ok := err.(*Error) | ||||
| 	if !ok { | ||||
| 		return Unspecified | ||||
| 	} | ||||
| 	if e.Code == Unspecified { | ||||
| 		return CanonicalCode(e.Err) | ||||
| 	} | ||||
| 	return e.Code | ||||
| } | ||||
|  | ||||
| /****************************************** | ||||
|     Domin Specific Error Types & Values | ||||
| *******************************************/ | ||||
|  | ||||
| var ( | ||||
| 	// ErrNoProcessableTask indicates that there are no tasks ready to be processed. | ||||
| 	ErrNoProcessableTask = errors.New("no tasks are ready for processing") | ||||
|  | ||||
| 	// ErrDuplicateTask indicates that another task with the same unique key holds the uniqueness lock. | ||||
| 	ErrDuplicateTask = errors.New("task already exists") | ||||
|  | ||||
| 	// ErrTaskIdConflict indicates that another task with the same task ID already exist | ||||
| 	ErrTaskIdConflict = errors.New("task id conflicts with another task") | ||||
| ) | ||||
|  | ||||
| // TaskNotFoundError indicates that a task with the given ID does not exist | ||||
| // in the given queue. | ||||
| type TaskNotFoundError struct { | ||||
| 	Queue string // queue name | ||||
| 	ID    string // task id | ||||
| } | ||||
|  | ||||
| func (e *TaskNotFoundError) Error() string { | ||||
| 	return fmt.Sprintf("cannot find task with id=%s in queue %q", e.ID, e.Queue) | ||||
| } | ||||
|  | ||||
| // IsTaskNotFound reports whether any error in err's chain is of type TaskNotFoundError. | ||||
| func IsTaskNotFound(err error) bool { | ||||
| 	var target *TaskNotFoundError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // QueueNotFoundError indicates that a queue with the given name does not exist. | ||||
| type QueueNotFoundError struct { | ||||
| 	Queue string // queue name | ||||
| } | ||||
|  | ||||
| func (e *QueueNotFoundError) Error() string { | ||||
| 	return fmt.Sprintf("queue %q does not exist", e.Queue) | ||||
| } | ||||
|  | ||||
| // IsQueueNotFound reports whether any error in err's chain is of type QueueNotFoundError. | ||||
| func IsQueueNotFound(err error) bool { | ||||
| 	var target *QueueNotFoundError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // QueueNotEmptyError indicates that the given queue is not empty. | ||||
| type QueueNotEmptyError struct { | ||||
| 	Queue string // queue name | ||||
| } | ||||
|  | ||||
| func (e *QueueNotEmptyError) Error() string { | ||||
| 	return fmt.Sprintf("queue %q is not empty", e.Queue) | ||||
| } | ||||
|  | ||||
| // IsQueueNotEmpty reports whether any error in err's chain is of type QueueNotEmptyError. | ||||
| func IsQueueNotEmpty(err error) bool { | ||||
| 	var target *QueueNotEmptyError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // TaskAlreadyArchivedError indicates that the task in question is already archived. | ||||
| type TaskAlreadyArchivedError struct { | ||||
| 	Queue string // queue name | ||||
| 	ID    string // task id | ||||
| } | ||||
|  | ||||
| func (e *TaskAlreadyArchivedError) Error() string { | ||||
| 	return fmt.Sprintf("task is already archived: id=%s, queue=%s", e.ID, e.Queue) | ||||
| } | ||||
|  | ||||
| // IsTaskAlreadyArchived reports whether any error in err's chain is of type TaskAlreadyArchivedError. | ||||
| func IsTaskAlreadyArchived(err error) bool { | ||||
| 	var target *TaskAlreadyArchivedError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| // RedisCommandError indicates that the given redis command returned error. | ||||
| type RedisCommandError struct { | ||||
| 	Command string // redis command (e.g. LRANGE, ZADD, etc) | ||||
| 	Err     error  // underlying error | ||||
| } | ||||
|  | ||||
| func (e *RedisCommandError) Error() string { | ||||
| 	return fmt.Sprintf("redis command error: %s failed: %v", strings.ToUpper(e.Command), e.Err) | ||||
| } | ||||
|  | ||||
| func (e *RedisCommandError) Unwrap() error { return e.Err } | ||||
|  | ||||
| // IsRedisCommandError reports whether any error in err's chain is of type RedisCommandError. | ||||
| func IsRedisCommandError(err error) bool { | ||||
| 	var target *RedisCommandError | ||||
| 	return As(err, &target) | ||||
| } | ||||
|  | ||||
| /************************************************* | ||||
|     Standard Library errors package functions | ||||
| *************************************************/ | ||||
|  | ||||
| // New returns an error that formats as the given text. | ||||
| // Each call to New returns a distinct error value even if the text is identical. | ||||
| // | ||||
| // This function is the errors.New function from the standard libarary (https://golang.org/pkg/errors/#New). | ||||
| // It is exported from this package for import convinience. | ||||
| func New(text string) error { return errors.New(text) } | ||||
|  | ||||
| // Is reports whether any error in err's chain matches target. | ||||
| // | ||||
| // This function is the errors.Is function from the standard libarary (https://golang.org/pkg/errors/#Is). | ||||
| // It is exported from this package for import convinience. | ||||
| func Is(err, target error) bool { return errors.Is(err, target) } | ||||
|  | ||||
| // As finds the first error in err's chain that matches target, and if so, sets target to that error value and returns true. | ||||
| // Otherwise, it returns false. | ||||
| // | ||||
| // This function is the errors.As function from the standard libarary (https://golang.org/pkg/errors/#As). | ||||
| // It is exported from this package for import convinience. | ||||
| func As(err error, target interface{}) bool { return errors.As(err, target) } | ||||
|  | ||||
| // Unwrap returns the result of calling the Unwrap method on err, if err's type contains an Unwrap method returning error. | ||||
| // Otherwise, Unwrap returns nil. | ||||
| // | ||||
| // This function is the errors.Unwrap function from the standard libarary (https://golang.org/pkg/errors/#Unwrap). | ||||
| // It is exported from this package for import convinience. | ||||
| func Unwrap(err error) error { return errors.Unwrap(err) } | ||||
							
								
								
									
										176
									
								
								internal/errors/errors_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								internal/errors/errors_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package errors | ||||
|  | ||||
| import "testing" | ||||
|  | ||||
| func TestErrorDebugString(t *testing.T) { | ||||
| 	// DebugString should include Op since its meant to be used by | ||||
| 	// maintainers/contributors of the asynq package. | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| 		err  error | ||||
| 		want string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "With Op, Code, and string", | ||||
| 			err:  E(Op("rdb.DeleteTask"), NotFound, "cannot find task with id=123"), | ||||
| 			want: "rdb.DeleteTask: NOT_FOUND: cannot find task with id=123", | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With Op, Code and error", | ||||
| 			err:  E(Op("rdb.DeleteTask"), NotFound, &TaskNotFoundError{Queue: "default", ID: "123"}), | ||||
| 			want: `rdb.DeleteTask: NOT_FOUND: cannot find task with id=123 in queue "default"`, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := tc.err.(*Error).DebugString(); got != tc.want { | ||||
| 			t.Errorf("%s: got=%q, want=%q", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorString(t *testing.T) { | ||||
| 	// String method should omit Op since op is an internal detail | ||||
| 	// and we don't want to provide it to users of the package. | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| 		err  error | ||||
| 		want string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "With Op, Code, and string", | ||||
| 			err:  E(Op("rdb.DeleteTask"), NotFound, "cannot find task with id=123"), | ||||
| 			want: "NOT_FOUND: cannot find task with id=123", | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "With Op, Code and error", | ||||
| 			err:  E(Op("rdb.DeleteTask"), NotFound, &TaskNotFoundError{Queue: "default", ID: "123"}), | ||||
| 			want: `NOT_FOUND: cannot find task with id=123 in queue "default"`, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := tc.err.Error(); got != tc.want { | ||||
| 			t.Errorf("%s: got=%q, want=%q", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorIs(t *testing.T) { | ||||
| 	var ErrCustom = New("custom sentinel error") | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc   string | ||||
| 		err    error | ||||
| 		target error | ||||
| 		want   bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:   "should unwrap one level", | ||||
| 			err:    E(Op("rdb.DeleteTask"), ErrCustom), | ||||
| 			target: ErrCustom, | ||||
| 			want:   true, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := Is(tc.err, tc.target); got != tc.want { | ||||
| 			t.Errorf("%s: got=%t, want=%t", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorAs(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc   string | ||||
| 		err    error | ||||
| 		target interface{} | ||||
| 		want   bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:   "should unwrap one level", | ||||
| 			err:    E(Op("rdb.DeleteTask"), NotFound, &QueueNotFoundError{Queue: "email"}), | ||||
| 			target: &QueueNotFoundError{}, | ||||
| 			want:   true, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := As(tc.err, &tc.target); got != tc.want { | ||||
| 			t.Errorf("%s: got=%t, want=%t", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorPredicates(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| 		fn   func(err error) bool | ||||
| 		err  error | ||||
| 		want bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "IsTaskNotFound should detect presence of TaskNotFoundError in err's chain", | ||||
| 			fn:   IsTaskNotFound, | ||||
| 			err:  E(Op("rdb.ArchiveTask"), NotFound, &TaskNotFoundError{Queue: "default", ID: "9876"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "IsTaskNotFound should detect absence of TaskNotFoundError in err's chain", | ||||
| 			fn:   IsTaskNotFound, | ||||
| 			err:  E(Op("rdb.ArchiveTask"), NotFound, &QueueNotFoundError{Queue: "default"}), | ||||
| 			want: false, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "IsQueueNotFound should detect presence of QueueNotFoundError in err's chain", | ||||
| 			fn:   IsQueueNotFound, | ||||
| 			err:  E(Op("rdb.ArchiveTask"), NotFound, &QueueNotFoundError{Queue: "default"}), | ||||
| 			want: true, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := tc.fn(tc.err); got != tc.want { | ||||
| 			t.Errorf("%s: got=%t, want=%t", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCanonicalCode(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc string | ||||
| 		err  error | ||||
| 		want Code | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc: "without nesting", | ||||
| 			err:  E(Op("rdb.DeleteTask"), NotFound, &TaskNotFoundError{Queue: "default", ID: "123"}), | ||||
| 			want: NotFound, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "with nesting", | ||||
| 			err:  E(FailedPrecondition, E(NotFound)), | ||||
| 			want: FailedPrecondition, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "returns Unspecified if err is not *Error", | ||||
| 			err:  New("some other error"), | ||||
| 			want: Unspecified, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc: "returns Unspecified if err is nil", | ||||
| 			err:  nil, | ||||
| 			want: Unspecified, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		if got := CanonicalCode(tc.err); got != tc.want { | ||||
| 			t.Errorf("%s: got=%s, want=%s", tc.desc, got, tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										838
									
								
								internal/proto/asynq.pb.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										838
									
								
								internal/proto/asynq.pb.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,838 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| // Code generated by protoc-gen-go. DO NOT EDIT. | ||||
| // versions: | ||||
| // 	protoc-gen-go v1.25.0 | ||||
| // 	protoc        v3.17.3 | ||||
| // 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" | ||||
| 	reflect "reflect" | ||||
| 	sync "sync" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	// Verify that this generated code is sufficiently up-to-date. | ||||
| 	_ = protoimpl.EnforceVersion(20 - protoimpl.MinVersion) | ||||
| 	// Verify that runtime/protoimpl is sufficiently up-to-date. | ||||
| 	_ = 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. | ||||
| type TaskMessage struct { | ||||
| 	state         protoimpl.MessageState | ||||
| 	sizeCache     protoimpl.SizeCache | ||||
| 	unknownFields protoimpl.UnknownFields | ||||
|  | ||||
| 	// Type indicates the kind of the task to be performed. | ||||
| 	Type string `protobuf:"bytes,1,opt,name=type,proto3" json:"type,omitempty"` | ||||
| 	// Payload holds data needed to process the task. | ||||
| 	Payload []byte `protobuf:"bytes,2,opt,name=payload,proto3" json:"payload,omitempty"` | ||||
| 	// Unique identifier for the task. | ||||
| 	Id string `protobuf:"bytes,3,opt,name=id,proto3" json:"id,omitempty"` | ||||
| 	// Name of the queue to which this task belongs. | ||||
| 	Queue string `protobuf:"bytes,4,opt,name=queue,proto3" json:"queue,omitempty"` | ||||
| 	// Max number of retries for this task. | ||||
| 	Retry int32 `protobuf:"varint,5,opt,name=retry,proto3" json:"retry,omitempty"` | ||||
| 	// Number of times this task has been retried so far. | ||||
| 	Retried int32 `protobuf:"varint,6,opt,name=retried,proto3" json:"retried,omitempty"` | ||||
| 	// Error message from the last failure. | ||||
| 	ErrorMsg string `protobuf:"bytes,7,opt,name=error_msg,json=errorMsg,proto3" json:"error_msg,omitempty"` | ||||
| 	// Time of last failure in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// Use zero to indicate no last failure. | ||||
| 	LastFailedAt int64 `protobuf:"varint,11,opt,name=last_failed_at,json=lastFailedAt,proto3" json:"last_failed_at,omitempty"` | ||||
| 	// Timeout specifies timeout in seconds. | ||||
| 	// Use zero to indicate no timeout. | ||||
| 	Timeout int64 `protobuf:"varint,8,opt,name=timeout,proto3" json:"timeout,omitempty"` | ||||
| 	// Deadline specifies the deadline for the task in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// Use zero to indicate no deadline. | ||||
| 	Deadline int64 `protobuf:"varint,9,opt,name=deadline,proto3" json:"deadline,omitempty"` | ||||
| 	// UniqueKey holds the redis key used for uniqueness lock for this task. | ||||
| 	// Empty string indicates that no uniqueness lock was used. | ||||
| 	UniqueKey string `protobuf:"bytes,10,opt,name=unique_key,json=uniqueKey,proto3" json:"unique_key,omitempty"` | ||||
| 	// Retention period specified in a number of seconds. | ||||
| 	// The task will be stored in redis as a completed task until the TTL | ||||
| 	// expires. | ||||
| 	Retention int64 `protobuf:"varint,12,opt,name=retention,proto3" json:"retention,omitempty"` | ||||
| 	// Time when the task completed in success in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// This field is populated if result_ttl > 0 upon completion. | ||||
| 	CompletedAt int64 `protobuf:"varint,13,opt,name=completed_at,json=completedAt,proto3" json:"completed_at,omitempty"` | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) Reset() { | ||||
| 	*x = TaskMessage{} | ||||
| 	if protoimpl.UnsafeEnabled { | ||||
| 		mi := &file_asynq_proto_msgTypes[0] | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		ms.StoreMessageInfo(mi) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) String() string { | ||||
| 	return protoimpl.X.MessageStringOf(x) | ||||
| } | ||||
|  | ||||
| func (*TaskMessage) ProtoMessage() {} | ||||
|  | ||||
| func (x *TaskMessage) ProtoReflect() protoreflect.Message { | ||||
| 	mi := &file_asynq_proto_msgTypes[0] | ||||
| 	if protoimpl.UnsafeEnabled && x != nil { | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		if ms.LoadMessageInfo() == nil { | ||||
| 			ms.StoreMessageInfo(mi) | ||||
| 		} | ||||
| 		return ms | ||||
| 	} | ||||
| 	return mi.MessageOf(x) | ||||
| } | ||||
|  | ||||
| // Deprecated: Use TaskMessage.ProtoReflect.Descriptor instead. | ||||
| func (*TaskMessage) Descriptor() ([]byte, []int) { | ||||
| 	return file_asynq_proto_rawDescGZIP(), []int{0} | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetType() string { | ||||
| 	if x != nil { | ||||
| 		return x.Type | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetPayload() []byte { | ||||
| 	if x != nil { | ||||
| 		return x.Payload | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetId() string { | ||||
| 	if x != nil { | ||||
| 		return x.Id | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetQueue() string { | ||||
| 	if x != nil { | ||||
| 		return x.Queue | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetRetry() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Retry | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetRetried() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Retried | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetErrorMsg() string { | ||||
| 	if x != nil { | ||||
| 		return x.ErrorMsg | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetLastFailedAt() int64 { | ||||
| 	if x != nil { | ||||
| 		return x.LastFailedAt | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetTimeout() int64 { | ||||
| 	if x != nil { | ||||
| 		return x.Timeout | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetDeadline() int64 { | ||||
| 	if x != nil { | ||||
| 		return x.Deadline | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetUniqueKey() string { | ||||
| 	if x != nil { | ||||
| 		return x.UniqueKey | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetRetention() int64 { | ||||
| 	if x != nil { | ||||
| 		return x.Retention | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *TaskMessage) GetCompletedAt() int64 { | ||||
| 	if x != nil { | ||||
| 		return x.CompletedAt | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| // ServerInfo holds information about a running server. | ||||
| type ServerInfo struct { | ||||
| 	state         protoimpl.MessageState | ||||
| 	sizeCache     protoimpl.SizeCache | ||||
| 	unknownFields protoimpl.UnknownFields | ||||
|  | ||||
| 	// Host machine the server is running on. | ||||
| 	Host string `protobuf:"bytes,1,opt,name=host,proto3" json:"host,omitempty"` | ||||
| 	// PID of the server process. | ||||
| 	Pid int32 `protobuf:"varint,2,opt,name=pid,proto3" json:"pid,omitempty"` | ||||
| 	// Unique identifier for this server. | ||||
| 	ServerId string `protobuf:"bytes,3,opt,name=server_id,json=serverId,proto3" json:"server_id,omitempty"` | ||||
| 	// Maximum number of concurrency this server will use. | ||||
| 	Concurrency int32 `protobuf:"varint,4,opt,name=concurrency,proto3" json:"concurrency,omitempty"` | ||||
| 	// List of queue names with their priorities. | ||||
| 	// The server will consume tasks from the queues and prioritize | ||||
| 	// queues with higher priority numbers. | ||||
| 	Queues map[string]int32 `protobuf:"bytes,5,rep,name=queues,proto3" json:"queues,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"varint,2,opt,name=value,proto3"` | ||||
| 	// If set, the server will always consume tasks from a queue with higher | ||||
| 	// priority. | ||||
| 	StrictPriority bool `protobuf:"varint,6,opt,name=strict_priority,json=strictPriority,proto3" json:"strict_priority,omitempty"` | ||||
| 	// Status indicates the status of the server. | ||||
| 	Status string `protobuf:"bytes,7,opt,name=status,proto3" json:"status,omitempty"` | ||||
| 	// Time this server was started. | ||||
| 	StartTime *timestamppb.Timestamp `protobuf:"bytes,8,opt,name=start_time,json=startTime,proto3" json:"start_time,omitempty"` | ||||
| 	// Number of workers currently processing tasks. | ||||
| 	ActiveWorkerCount int32 `protobuf:"varint,9,opt,name=active_worker_count,json=activeWorkerCount,proto3" json:"active_worker_count,omitempty"` | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) Reset() { | ||||
| 	*x = ServerInfo{} | ||||
| 	if protoimpl.UnsafeEnabled { | ||||
| 		mi := &file_asynq_proto_msgTypes[1] | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		ms.StoreMessageInfo(mi) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) String() string { | ||||
| 	return protoimpl.X.MessageStringOf(x) | ||||
| } | ||||
|  | ||||
| func (*ServerInfo) ProtoMessage() {} | ||||
|  | ||||
| func (x *ServerInfo) ProtoReflect() protoreflect.Message { | ||||
| 	mi := &file_asynq_proto_msgTypes[1] | ||||
| 	if protoimpl.UnsafeEnabled && x != nil { | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		if ms.LoadMessageInfo() == nil { | ||||
| 			ms.StoreMessageInfo(mi) | ||||
| 		} | ||||
| 		return ms | ||||
| 	} | ||||
| 	return mi.MessageOf(x) | ||||
| } | ||||
|  | ||||
| // Deprecated: Use ServerInfo.ProtoReflect.Descriptor instead. | ||||
| func (*ServerInfo) Descriptor() ([]byte, []int) { | ||||
| 	return file_asynq_proto_rawDescGZIP(), []int{1} | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetHost() string { | ||||
| 	if x != nil { | ||||
| 		return x.Host | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetPid() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Pid | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetServerId() string { | ||||
| 	if x != nil { | ||||
| 		return x.ServerId | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetConcurrency() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Concurrency | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetQueues() map[string]int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Queues | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetStrictPriority() bool { | ||||
| 	if x != nil { | ||||
| 		return x.StrictPriority | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetStatus() string { | ||||
| 	if x != nil { | ||||
| 		return x.Status | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetStartTime() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.StartTime | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *ServerInfo) GetActiveWorkerCount() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.ActiveWorkerCount | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| // WorkerInfo holds information about a running worker. | ||||
| type WorkerInfo struct { | ||||
| 	state         protoimpl.MessageState | ||||
| 	sizeCache     protoimpl.SizeCache | ||||
| 	unknownFields protoimpl.UnknownFields | ||||
|  | ||||
| 	// Host matchine this worker is running on. | ||||
| 	Host string `protobuf:"bytes,1,opt,name=host,proto3" json:"host,omitempty"` | ||||
| 	// PID of the process in which this worker is running. | ||||
| 	Pid int32 `protobuf:"varint,2,opt,name=pid,proto3" json:"pid,omitempty"` | ||||
| 	// ID of the server in which this worker is running. | ||||
| 	ServerId string `protobuf:"bytes,3,opt,name=server_id,json=serverId,proto3" json:"server_id,omitempty"` | ||||
| 	// ID of the task this worker is processing. | ||||
| 	TaskId string `protobuf:"bytes,4,opt,name=task_id,json=taskId,proto3" json:"task_id,omitempty"` | ||||
| 	// Type of the task this worker is processing. | ||||
| 	TaskType string `protobuf:"bytes,5,opt,name=task_type,json=taskType,proto3" json:"task_type,omitempty"` | ||||
| 	// Payload of the task this worker is processing. | ||||
| 	TaskPayload []byte `protobuf:"bytes,6,opt,name=task_payload,json=taskPayload,proto3" json:"task_payload,omitempty"` | ||||
| 	// Name of the queue the task the worker is processing belongs. | ||||
| 	Queue string `protobuf:"bytes,7,opt,name=queue,proto3" json:"queue,omitempty"` | ||||
| 	// Time this worker started processing the task. | ||||
| 	StartTime *timestamppb.Timestamp `protobuf:"bytes,8,opt,name=start_time,json=startTime,proto3" json:"start_time,omitempty"` | ||||
| 	// Deadline by which the worker needs to complete processing | ||||
| 	// the task. If worker exceeds the deadline, the task will fail. | ||||
| 	Deadline *timestamppb.Timestamp `protobuf:"bytes,9,opt,name=deadline,proto3" json:"deadline,omitempty"` | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) Reset() { | ||||
| 	*x = WorkerInfo{} | ||||
| 	if protoimpl.UnsafeEnabled { | ||||
| 		mi := &file_asynq_proto_msgTypes[2] | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		ms.StoreMessageInfo(mi) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) String() string { | ||||
| 	return protoimpl.X.MessageStringOf(x) | ||||
| } | ||||
|  | ||||
| func (*WorkerInfo) ProtoMessage() {} | ||||
|  | ||||
| func (x *WorkerInfo) ProtoReflect() protoreflect.Message { | ||||
| 	mi := &file_asynq_proto_msgTypes[2] | ||||
| 	if protoimpl.UnsafeEnabled && x != nil { | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		if ms.LoadMessageInfo() == nil { | ||||
| 			ms.StoreMessageInfo(mi) | ||||
| 		} | ||||
| 		return ms | ||||
| 	} | ||||
| 	return mi.MessageOf(x) | ||||
| } | ||||
|  | ||||
| // Deprecated: Use WorkerInfo.ProtoReflect.Descriptor instead. | ||||
| func (*WorkerInfo) Descriptor() ([]byte, []int) { | ||||
| 	return file_asynq_proto_rawDescGZIP(), []int{2} | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetHost() string { | ||||
| 	if x != nil { | ||||
| 		return x.Host | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetPid() int32 { | ||||
| 	if x != nil { | ||||
| 		return x.Pid | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetServerId() string { | ||||
| 	if x != nil { | ||||
| 		return x.ServerId | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetTaskId() string { | ||||
| 	if x != nil { | ||||
| 		return x.TaskId | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetTaskType() string { | ||||
| 	if x != nil { | ||||
| 		return x.TaskType | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetTaskPayload() []byte { | ||||
| 	if x != nil { | ||||
| 		return x.TaskPayload | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetQueue() string { | ||||
| 	if x != nil { | ||||
| 		return x.Queue | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetStartTime() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.StartTime | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *WorkerInfo) GetDeadline() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.Deadline | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SchedulerEntry holds information about a periodic task registered | ||||
| // with a scheduler. | ||||
| type SchedulerEntry struct { | ||||
| 	state         protoimpl.MessageState | ||||
| 	sizeCache     protoimpl.SizeCache | ||||
| 	unknownFields protoimpl.UnknownFields | ||||
|  | ||||
| 	// Identifier of the scheduler entry. | ||||
| 	Id string `protobuf:"bytes,1,opt,name=id,proto3" json:"id,omitempty"` | ||||
| 	// Periodic schedule spec of the entry. | ||||
| 	Spec string `protobuf:"bytes,2,opt,name=spec,proto3" json:"spec,omitempty"` | ||||
| 	// Task type of the periodic task. | ||||
| 	TaskType string `protobuf:"bytes,3,opt,name=task_type,json=taskType,proto3" json:"task_type,omitempty"` | ||||
| 	// Task payload of the periodic task. | ||||
| 	TaskPayload []byte `protobuf:"bytes,4,opt,name=task_payload,json=taskPayload,proto3" json:"task_payload,omitempty"` | ||||
| 	// Options used to enqueue the periodic task. | ||||
| 	EnqueueOptions []string `protobuf:"bytes,5,rep,name=enqueue_options,json=enqueueOptions,proto3" json:"enqueue_options,omitempty"` | ||||
| 	// Next time the task will be enqueued. | ||||
| 	NextEnqueueTime *timestamppb.Timestamp `protobuf:"bytes,6,opt,name=next_enqueue_time,json=nextEnqueueTime,proto3" json:"next_enqueue_time,omitempty"` | ||||
| 	// Last time the task was enqueued. | ||||
| 	// Zero time if task was never enqueued. | ||||
| 	PrevEnqueueTime *timestamppb.Timestamp `protobuf:"bytes,7,opt,name=prev_enqueue_time,json=prevEnqueueTime,proto3" json:"prev_enqueue_time,omitempty"` | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) Reset() { | ||||
| 	*x = SchedulerEntry{} | ||||
| 	if protoimpl.UnsafeEnabled { | ||||
| 		mi := &file_asynq_proto_msgTypes[3] | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		ms.StoreMessageInfo(mi) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) String() string { | ||||
| 	return protoimpl.X.MessageStringOf(x) | ||||
| } | ||||
|  | ||||
| func (*SchedulerEntry) ProtoMessage() {} | ||||
|  | ||||
| func (x *SchedulerEntry) ProtoReflect() protoreflect.Message { | ||||
| 	mi := &file_asynq_proto_msgTypes[3] | ||||
| 	if protoimpl.UnsafeEnabled && x != nil { | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		if ms.LoadMessageInfo() == nil { | ||||
| 			ms.StoreMessageInfo(mi) | ||||
| 		} | ||||
| 		return ms | ||||
| 	} | ||||
| 	return mi.MessageOf(x) | ||||
| } | ||||
|  | ||||
| // Deprecated: Use SchedulerEntry.ProtoReflect.Descriptor instead. | ||||
| func (*SchedulerEntry) Descriptor() ([]byte, []int) { | ||||
| 	return file_asynq_proto_rawDescGZIP(), []int{3} | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetId() string { | ||||
| 	if x != nil { | ||||
| 		return x.Id | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetSpec() string { | ||||
| 	if x != nil { | ||||
| 		return x.Spec | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetTaskType() string { | ||||
| 	if x != nil { | ||||
| 		return x.TaskType | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetTaskPayload() []byte { | ||||
| 	if x != nil { | ||||
| 		return x.TaskPayload | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetEnqueueOptions() []string { | ||||
| 	if x != nil { | ||||
| 		return x.EnqueueOptions | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetNextEnqueueTime() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.NextEnqueueTime | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEntry) GetPrevEnqueueTime() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.PrevEnqueueTime | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SchedulerEnqueueEvent holds information about an enqueue event | ||||
| // by a scheduler. | ||||
| type SchedulerEnqueueEvent struct { | ||||
| 	state         protoimpl.MessageState | ||||
| 	sizeCache     protoimpl.SizeCache | ||||
| 	unknownFields protoimpl.UnknownFields | ||||
|  | ||||
| 	// ID of the task that was enqueued. | ||||
| 	TaskId string `protobuf:"bytes,1,opt,name=task_id,json=taskId,proto3" json:"task_id,omitempty"` | ||||
| 	// Time the task was enqueued. | ||||
| 	EnqueueTime *timestamppb.Timestamp `protobuf:"bytes,2,opt,name=enqueue_time,json=enqueueTime,proto3" json:"enqueue_time,omitempty"` | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEnqueueEvent) Reset() { | ||||
| 	*x = SchedulerEnqueueEvent{} | ||||
| 	if protoimpl.UnsafeEnabled { | ||||
| 		mi := &file_asynq_proto_msgTypes[4] | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		ms.StoreMessageInfo(mi) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEnqueueEvent) String() string { | ||||
| 	return protoimpl.X.MessageStringOf(x) | ||||
| } | ||||
|  | ||||
| func (*SchedulerEnqueueEvent) ProtoMessage() {} | ||||
|  | ||||
| func (x *SchedulerEnqueueEvent) ProtoReflect() protoreflect.Message { | ||||
| 	mi := &file_asynq_proto_msgTypes[4] | ||||
| 	if protoimpl.UnsafeEnabled && x != nil { | ||||
| 		ms := protoimpl.X.MessageStateOf(protoimpl.Pointer(x)) | ||||
| 		if ms.LoadMessageInfo() == nil { | ||||
| 			ms.StoreMessageInfo(mi) | ||||
| 		} | ||||
| 		return ms | ||||
| 	} | ||||
| 	return mi.MessageOf(x) | ||||
| } | ||||
|  | ||||
| // Deprecated: Use SchedulerEnqueueEvent.ProtoReflect.Descriptor instead. | ||||
| func (*SchedulerEnqueueEvent) Descriptor() ([]byte, []int) { | ||||
| 	return file_asynq_proto_rawDescGZIP(), []int{4} | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEnqueueEvent) GetTaskId() string { | ||||
| 	if x != nil { | ||||
| 		return x.TaskId | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (x *SchedulerEnqueueEvent) GetEnqueueTime() *timestamppb.Timestamp { | ||||
| 	if x != nil { | ||||
| 		return x.EnqueueTime | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| var File_asynq_proto protoreflect.FileDescriptor | ||||
|  | ||||
| var file_asynq_proto_rawDesc = []byte{ | ||||
| 	0x0a, 0x0b, 0x61, 0x73, 0x79, 0x6e, 0x71, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x12, 0x05, 0x61, | ||||
| 	0x73, 0x79, 0x6e, 0x71, 0x1a, 0x1f, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2f, 0x70, 0x72, 0x6f, | ||||
| 	0x74, 0x6f, 0x62, 0x75, 0x66, 0x2f, 0x74, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x2e, | ||||
| 	0x70, 0x72, 0x6f, 0x74, 0x6f, 0x22, 0xea, 0x02, 0x0a, 0x0b, 0x54, 0x61, 0x73, 0x6b, 0x4d, 0x65, | ||||
| 	0x73, 0x73, 0x61, 0x67, 0x65, 0x12, 0x12, 0x0a, 0x04, 0x74, 0x79, 0x70, 0x65, 0x18, 0x01, 0x20, | ||||
| 	0x01, 0x28, 0x09, 0x52, 0x04, 0x74, 0x79, 0x70, 0x65, 0x12, 0x18, 0x0a, 0x07, 0x70, 0x61, 0x79, | ||||
| 	0x6c, 0x6f, 0x61, 0x64, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0c, 0x52, 0x07, 0x70, 0x61, 0x79, 0x6c, | ||||
| 	0x6f, 0x61, 0x64, 0x12, 0x0e, 0x0a, 0x02, 0x69, 0x64, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, | ||||
| 	0x02, 0x69, 0x64, 0x12, 0x14, 0x0a, 0x05, 0x71, 0x75, 0x65, 0x75, 0x65, 0x18, 0x04, 0x20, 0x01, | ||||
| 	0x28, 0x09, 0x52, 0x05, 0x71, 0x75, 0x65, 0x75, 0x65, 0x12, 0x14, 0x0a, 0x05, 0x72, 0x65, 0x74, | ||||
| 	0x72, 0x79, 0x18, 0x05, 0x20, 0x01, 0x28, 0x05, 0x52, 0x05, 0x72, 0x65, 0x74, 0x72, 0x79, 0x12, | ||||
| 	0x18, 0x0a, 0x07, 0x72, 0x65, 0x74, 0x72, 0x69, 0x65, 0x64, 0x18, 0x06, 0x20, 0x01, 0x28, 0x05, | ||||
| 	0x52, 0x07, 0x72, 0x65, 0x74, 0x72, 0x69, 0x65, 0x64, 0x12, 0x1b, 0x0a, 0x09, 0x65, 0x72, 0x72, | ||||
| 	0x6f, 0x72, 0x5f, 0x6d, 0x73, 0x67, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x65, 0x72, | ||||
| 	0x72, 0x6f, 0x72, 0x4d, 0x73, 0x67, 0x12, 0x24, 0x0a, 0x0e, 0x6c, 0x61, 0x73, 0x74, 0x5f, 0x66, | ||||
| 	0x61, 0x69, 0x6c, 0x65, 0x64, 0x5f, 0x61, 0x74, 0x18, 0x0b, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0c, | ||||
| 	0x6c, 0x61, 0x73, 0x74, 0x46, 0x61, 0x69, 0x6c, 0x65, 0x64, 0x41, 0x74, 0x12, 0x18, 0x0a, 0x07, | ||||
| 	0x74, 0x69, 0x6d, 0x65, 0x6f, 0x75, 0x74, 0x18, 0x08, 0x20, 0x01, 0x28, 0x03, 0x52, 0x07, 0x74, | ||||
| 	0x69, 0x6d, 0x65, 0x6f, 0x75, 0x74, 0x12, 0x1a, 0x0a, 0x08, 0x64, 0x65, 0x61, 0x64, 0x6c, 0x69, | ||||
| 	0x6e, 0x65, 0x18, 0x09, 0x20, 0x01, 0x28, 0x03, 0x52, 0x08, 0x64, 0x65, 0x61, 0x64, 0x6c, 0x69, | ||||
| 	0x6e, 0x65, 0x12, 0x1d, 0x0a, 0x0a, 0x75, 0x6e, 0x69, 0x71, 0x75, 0x65, 0x5f, 0x6b, 0x65, 0x79, | ||||
| 	0x18, 0x0a, 0x20, 0x01, 0x28, 0x09, 0x52, 0x09, 0x75, 0x6e, 0x69, 0x71, 0x75, 0x65, 0x4b, 0x65, | ||||
| 	0x79, 0x12, 0x1c, 0x0a, 0x09, 0x72, 0x65, 0x74, 0x65, 0x6e, 0x74, 0x69, 0x6f, 0x6e, 0x18, 0x0c, | ||||
| 	0x20, 0x01, 0x28, 0x03, 0x52, 0x09, 0x72, 0x65, 0x74, 0x65, 0x6e, 0x74, 0x69, 0x6f, 0x6e, 0x12, | ||||
| 	0x21, 0x0a, 0x0c, 0x63, 0x6f, 0x6d, 0x70, 0x6c, 0x65, 0x74, 0x65, 0x64, 0x5f, 0x61, 0x74, 0x18, | ||||
| 	0x0d, 0x20, 0x01, 0x28, 0x03, 0x52, 0x0b, 0x63, 0x6f, 0x6d, 0x70, 0x6c, 0x65, 0x74, 0x65, 0x64, | ||||
| 	0x41, 0x74, 0x22, 0x8f, 0x03, 0x0a, 0x0a, 0x53, 0x65, 0x72, 0x76, 0x65, 0x72, 0x49, 0x6e, 0x66, | ||||
| 	0x6f, 0x12, 0x12, 0x0a, 0x04, 0x68, 0x6f, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, | ||||
| 	0x04, 0x68, 0x6f, 0x73, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x70, 0x69, 0x64, 0x18, 0x02, 0x20, 0x01, | ||||
| 	0x28, 0x05, 0x52, 0x03, 0x70, 0x69, 0x64, 0x12, 0x1b, 0x0a, 0x09, 0x73, 0x65, 0x72, 0x76, 0x65, | ||||
| 	0x72, 0x5f, 0x69, 0x64, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x73, 0x65, 0x72, 0x76, | ||||
| 	0x65, 0x72, 0x49, 0x64, 0x12, 0x20, 0x0a, 0x0b, 0x63, 0x6f, 0x6e, 0x63, 0x75, 0x72, 0x72, 0x65, | ||||
| 	0x6e, 0x63, 0x79, 0x18, 0x04, 0x20, 0x01, 0x28, 0x05, 0x52, 0x0b, 0x63, 0x6f, 0x6e, 0x63, 0x75, | ||||
| 	0x72, 0x72, 0x65, 0x6e, 0x63, 0x79, 0x12, 0x35, 0x0a, 0x06, 0x71, 0x75, 0x65, 0x75, 0x65, 0x73, | ||||
| 	0x18, 0x05, 0x20, 0x03, 0x28, 0x0b, 0x32, 0x1d, 0x2e, 0x61, 0x73, 0x79, 0x6e, 0x71, 0x2e, 0x53, | ||||
| 	0x65, 0x72, 0x76, 0x65, 0x72, 0x49, 0x6e, 0x66, 0x6f, 0x2e, 0x51, 0x75, 0x65, 0x75, 0x65, 0x73, | ||||
| 	0x45, 0x6e, 0x74, 0x72, 0x79, 0x52, 0x06, 0x71, 0x75, 0x65, 0x75, 0x65, 0x73, 0x12, 0x27, 0x0a, | ||||
| 	0x0f, 0x73, 0x74, 0x72, 0x69, 0x63, 0x74, 0x5f, 0x70, 0x72, 0x69, 0x6f, 0x72, 0x69, 0x74, 0x79, | ||||
| 	0x18, 0x06, 0x20, 0x01, 0x28, 0x08, 0x52, 0x0e, 0x73, 0x74, 0x72, 0x69, 0x63, 0x74, 0x50, 0x72, | ||||
| 	0x69, 0x6f, 0x72, 0x69, 0x74, 0x79, 0x12, 0x16, 0x0a, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, | ||||
| 	0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x73, 0x74, 0x61, 0x74, 0x75, 0x73, 0x12, 0x39, | ||||
| 	0x0a, 0x0a, 0x73, 0x74, 0x61, 0x72, 0x74, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x08, 0x20, 0x01, | ||||
| 	0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, | ||||
| 	0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x09, | ||||
| 	0x73, 0x74, 0x61, 0x72, 0x74, 0x54, 0x69, 0x6d, 0x65, 0x12, 0x2e, 0x0a, 0x13, 0x61, 0x63, 0x74, | ||||
| 	0x69, 0x76, 0x65, 0x5f, 0x77, 0x6f, 0x72, 0x6b, 0x65, 0x72, 0x5f, 0x63, 0x6f, 0x75, 0x6e, 0x74, | ||||
| 	0x18, 0x09, 0x20, 0x01, 0x28, 0x05, 0x52, 0x11, 0x61, 0x63, 0x74, 0x69, 0x76, 0x65, 0x57, 0x6f, | ||||
| 	0x72, 0x6b, 0x65, 0x72, 0x43, 0x6f, 0x75, 0x6e, 0x74, 0x1a, 0x39, 0x0a, 0x0b, 0x51, 0x75, 0x65, | ||||
| 	0x75, 0x65, 0x73, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x10, 0x0a, 0x03, 0x6b, 0x65, 0x79, 0x18, | ||||
| 	0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x03, 0x6b, 0x65, 0x79, 0x12, 0x14, 0x0a, 0x05, 0x76, 0x61, | ||||
| 	0x6c, 0x75, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x05, 0x52, 0x05, 0x76, 0x61, 0x6c, 0x75, 0x65, | ||||
| 	0x3a, 0x02, 0x38, 0x01, 0x22, 0xb1, 0x02, 0x0a, 0x0a, 0x57, 0x6f, 0x72, 0x6b, 0x65, 0x72, 0x49, | ||||
| 	0x6e, 0x66, 0x6f, 0x12, 0x12, 0x0a, 0x04, 0x68, 0x6f, 0x73, 0x74, 0x18, 0x01, 0x20, 0x01, 0x28, | ||||
| 	0x09, 0x52, 0x04, 0x68, 0x6f, 0x73, 0x74, 0x12, 0x10, 0x0a, 0x03, 0x70, 0x69, 0x64, 0x18, 0x02, | ||||
| 	0x20, 0x01, 0x28, 0x05, 0x52, 0x03, 0x70, 0x69, 0x64, 0x12, 0x1b, 0x0a, 0x09, 0x73, 0x65, 0x72, | ||||
| 	0x76, 0x65, 0x72, 0x5f, 0x69, 0x64, 0x18, 0x03, 0x20, 0x01, 0x28, 0x09, 0x52, 0x08, 0x73, 0x65, | ||||
| 	0x72, 0x76, 0x65, 0x72, 0x49, 0x64, 0x12, 0x17, 0x0a, 0x07, 0x74, 0x61, 0x73, 0x6b, 0x5f, 0x69, | ||||
| 	0x64, 0x18, 0x04, 0x20, 0x01, 0x28, 0x09, 0x52, 0x06, 0x74, 0x61, 0x73, 0x6b, 0x49, 0x64, 0x12, | ||||
| 	0x1b, 0x0a, 0x09, 0x74, 0x61, 0x73, 0x6b, 0x5f, 0x74, 0x79, 0x70, 0x65, 0x18, 0x05, 0x20, 0x01, | ||||
| 	0x28, 0x09, 0x52, 0x08, 0x74, 0x61, 0x73, 0x6b, 0x54, 0x79, 0x70, 0x65, 0x12, 0x21, 0x0a, 0x0c, | ||||
| 	0x74, 0x61, 0x73, 0x6b, 0x5f, 0x70, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x18, 0x06, 0x20, 0x01, | ||||
| 	0x28, 0x0c, 0x52, 0x0b, 0x74, 0x61, 0x73, 0x6b, 0x50, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x12, | ||||
| 	0x14, 0x0a, 0x05, 0x71, 0x75, 0x65, 0x75, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x09, 0x52, 0x05, | ||||
| 	0x71, 0x75, 0x65, 0x75, 0x65, 0x12, 0x39, 0x0a, 0x0a, 0x73, 0x74, 0x61, 0x72, 0x74, 0x5f, 0x74, | ||||
| 	0x69, 0x6d, 0x65, 0x18, 0x08, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, | ||||
| 	0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, | ||||
| 	0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x09, 0x73, 0x74, 0x61, 0x72, 0x74, 0x54, 0x69, 0x6d, 0x65, | ||||
| 	0x12, 0x36, 0x0a, 0x08, 0x64, 0x65, 0x61, 0x64, 0x6c, 0x69, 0x6e, 0x65, 0x18, 0x09, 0x20, 0x01, | ||||
| 	0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, | ||||
| 	0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x08, | ||||
| 	0x64, 0x65, 0x61, 0x64, 0x6c, 0x69, 0x6e, 0x65, 0x22, 0xad, 0x02, 0x0a, 0x0e, 0x53, 0x63, 0x68, | ||||
| 	0x65, 0x64, 0x75, 0x6c, 0x65, 0x72, 0x45, 0x6e, 0x74, 0x72, 0x79, 0x12, 0x0e, 0x0a, 0x02, 0x69, | ||||
| 	0x64, 0x18, 0x01, 0x20, 0x01, 0x28, 0x09, 0x52, 0x02, 0x69, 0x64, 0x12, 0x12, 0x0a, 0x04, 0x73, | ||||
| 	0x70, 0x65, 0x63, 0x18, 0x02, 0x20, 0x01, 0x28, 0x09, 0x52, 0x04, 0x73, 0x70, 0x65, 0x63, 0x12, | ||||
| 	0x1b, 0x0a, 0x09, 0x74, 0x61, 0x73, 0x6b, 0x5f, 0x74, 0x79, 0x70, 0x65, 0x18, 0x03, 0x20, 0x01, | ||||
| 	0x28, 0x09, 0x52, 0x08, 0x74, 0x61, 0x73, 0x6b, 0x54, 0x79, 0x70, 0x65, 0x12, 0x21, 0x0a, 0x0c, | ||||
| 	0x74, 0x61, 0x73, 0x6b, 0x5f, 0x70, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x18, 0x04, 0x20, 0x01, | ||||
| 	0x28, 0x0c, 0x52, 0x0b, 0x74, 0x61, 0x73, 0x6b, 0x50, 0x61, 0x79, 0x6c, 0x6f, 0x61, 0x64, 0x12, | ||||
| 	0x27, 0x0a, 0x0f, 0x65, 0x6e, 0x71, 0x75, 0x65, 0x75, 0x65, 0x5f, 0x6f, 0x70, 0x74, 0x69, 0x6f, | ||||
| 	0x6e, 0x73, 0x18, 0x05, 0x20, 0x03, 0x28, 0x09, 0x52, 0x0e, 0x65, 0x6e, 0x71, 0x75, 0x65, 0x75, | ||||
| 	0x65, 0x4f, 0x70, 0x74, 0x69, 0x6f, 0x6e, 0x73, 0x12, 0x46, 0x0a, 0x11, 0x6e, 0x65, 0x78, 0x74, | ||||
| 	0x5f, 0x65, 0x6e, 0x71, 0x75, 0x65, 0x75, 0x65, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x06, 0x20, | ||||
| 	0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, | ||||
| 	0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, | ||||
| 	0x0f, 0x6e, 0x65, 0x78, 0x74, 0x45, 0x6e, 0x71, 0x75, 0x65, 0x75, 0x65, 0x54, 0x69, 0x6d, 0x65, | ||||
| 	0x12, 0x46, 0x0a, 0x11, 0x70, 0x72, 0x65, 0x76, 0x5f, 0x65, 0x6e, 0x71, 0x75, 0x65, 0x75, 0x65, | ||||
| 	0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x07, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x1a, 0x2e, 0x67, 0x6f, | ||||
| 	0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x75, 0x66, 0x2e, 0x54, 0x69, | ||||
| 	0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x0f, 0x70, 0x72, 0x65, 0x76, 0x45, 0x6e, 0x71, | ||||
| 	0x75, 0x65, 0x75, 0x65, 0x54, 0x69, 0x6d, 0x65, 0x22, 0x6f, 0x0a, 0x15, 0x53, 0x63, 0x68, 0x65, | ||||
| 	0x64, 0x75, 0x6c, 0x65, 0x72, 0x45, 0x6e, 0x71, 0x75, 0x65, 0x75, 0x65, 0x45, 0x76, 0x65, 0x6e, | ||||
| 	0x74, 0x12, 0x17, 0x0a, 0x07, 0x74, 0x61, 0x73, 0x6b, 0x5f, 0x69, 0x64, 0x18, 0x01, 0x20, 0x01, | ||||
| 	0x28, 0x09, 0x52, 0x06, 0x74, 0x61, 0x73, 0x6b, 0x49, 0x64, 0x12, 0x3d, 0x0a, 0x0c, 0x65, 0x6e, | ||||
| 	0x71, 0x75, 0x65, 0x75, 0x65, 0x5f, 0x74, 0x69, 0x6d, 0x65, 0x18, 0x02, 0x20, 0x01, 0x28, 0x0b, | ||||
| 	0x32, 0x1a, 0x2e, 0x67, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, | ||||
| 	0x75, 0x66, 0x2e, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x74, 0x61, 0x6d, 0x70, 0x52, 0x0b, 0x65, 0x6e, | ||||
| 	0x71, 0x75, 0x65, 0x75, 0x65, 0x54, 0x69, 0x6d, 0x65, 0x42, 0x29, 0x5a, 0x27, 0x67, 0x69, 0x74, | ||||
| 	0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x68, 0x69, 0x62, 0x69, 0x6b, 0x65, 0x6e, 0x2f, | ||||
| 	0x61, 0x73, 0x79, 0x6e, 0x71, 0x2f, 0x69, 0x6e, 0x74, 0x65, 0x72, 0x6e, 0x61, 0x6c, 0x2f, 0x70, | ||||
| 	0x72, 0x6f, 0x74, 0x6f, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	file_asynq_proto_rawDescOnce sync.Once | ||||
| 	file_asynq_proto_rawDescData = file_asynq_proto_rawDesc | ||||
| ) | ||||
|  | ||||
| func file_asynq_proto_rawDescGZIP() []byte { | ||||
| 	file_asynq_proto_rawDescOnce.Do(func() { | ||||
| 		file_asynq_proto_rawDescData = protoimpl.X.CompressGZIP(file_asynq_proto_rawDescData) | ||||
| 	}) | ||||
| 	return file_asynq_proto_rawDescData | ||||
| } | ||||
|  | ||||
| var file_asynq_proto_msgTypes = make([]protoimpl.MessageInfo, 6) | ||||
| var file_asynq_proto_goTypes = []interface{}{ | ||||
| 	(*TaskMessage)(nil),           // 0: asynq.TaskMessage | ||||
| 	(*ServerInfo)(nil),            // 1: asynq.ServerInfo | ||||
| 	(*WorkerInfo)(nil),            // 2: asynq.WorkerInfo | ||||
| 	(*SchedulerEntry)(nil),        // 3: asynq.SchedulerEntry | ||||
| 	(*SchedulerEnqueueEvent)(nil), // 4: asynq.SchedulerEnqueueEvent | ||||
| 	nil,                           // 5: asynq.ServerInfo.QueuesEntry | ||||
| 	(*timestamppb.Timestamp)(nil), // 6: google.protobuf.Timestamp | ||||
| } | ||||
| var file_asynq_proto_depIdxs = []int32{ | ||||
| 	5, // 0: asynq.ServerInfo.queues:type_name -> asynq.ServerInfo.QueuesEntry | ||||
| 	6, // 1: asynq.ServerInfo.start_time:type_name -> google.protobuf.Timestamp | ||||
| 	6, // 2: asynq.WorkerInfo.start_time:type_name -> google.protobuf.Timestamp | ||||
| 	6, // 3: asynq.WorkerInfo.deadline:type_name -> google.protobuf.Timestamp | ||||
| 	6, // 4: asynq.SchedulerEntry.next_enqueue_time:type_name -> google.protobuf.Timestamp | ||||
| 	6, // 5: asynq.SchedulerEntry.prev_enqueue_time:type_name -> google.protobuf.Timestamp | ||||
| 	6, // 6: asynq.SchedulerEnqueueEvent.enqueue_time:type_name -> google.protobuf.Timestamp | ||||
| 	7, // [7:7] is the sub-list for method output_type | ||||
| 	7, // [7:7] is the sub-list for method input_type | ||||
| 	7, // [7:7] is the sub-list for extension type_name | ||||
| 	7, // [7:7] is the sub-list for extension extendee | ||||
| 	0, // [0:7] is the sub-list for field type_name | ||||
| } | ||||
|  | ||||
| func init() { file_asynq_proto_init() } | ||||
| func file_asynq_proto_init() { | ||||
| 	if File_asynq_proto != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	if !protoimpl.UnsafeEnabled { | ||||
| 		file_asynq_proto_msgTypes[0].Exporter = func(v interface{}, i int) interface{} { | ||||
| 			switch v := v.(*TaskMessage); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| 			case 1: | ||||
| 				return &v.sizeCache | ||||
| 			case 2: | ||||
| 				return &v.unknownFields | ||||
| 			default: | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[1].Exporter = func(v interface{}, i int) interface{} { | ||||
| 			switch v := v.(*ServerInfo); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| 			case 1: | ||||
| 				return &v.sizeCache | ||||
| 			case 2: | ||||
| 				return &v.unknownFields | ||||
| 			default: | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[2].Exporter = func(v interface{}, i int) interface{} { | ||||
| 			switch v := v.(*WorkerInfo); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| 			case 1: | ||||
| 				return &v.sizeCache | ||||
| 			case 2: | ||||
| 				return &v.unknownFields | ||||
| 			default: | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[3].Exporter = func(v interface{}, i int) interface{} { | ||||
| 			switch v := v.(*SchedulerEntry); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| 			case 1: | ||||
| 				return &v.sizeCache | ||||
| 			case 2: | ||||
| 				return &v.unknownFields | ||||
| 			default: | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		file_asynq_proto_msgTypes[4].Exporter = func(v interface{}, i int) interface{} { | ||||
| 			switch v := v.(*SchedulerEnqueueEvent); i { | ||||
| 			case 0: | ||||
| 				return &v.state | ||||
| 			case 1: | ||||
| 				return &v.sizeCache | ||||
| 			case 2: | ||||
| 				return &v.unknownFields | ||||
| 			default: | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	type x struct{} | ||||
| 	out := protoimpl.TypeBuilder{ | ||||
| 		File: protoimpl.DescBuilder{ | ||||
| 			GoPackagePath: reflect.TypeOf(x{}).PkgPath(), | ||||
| 			RawDescriptor: file_asynq_proto_rawDesc, | ||||
| 			NumEnums:      0, | ||||
| 			NumMessages:   6, | ||||
| 			NumExtensions: 0, | ||||
| 			NumServices:   0, | ||||
| 		}, | ||||
| 		GoTypes:           file_asynq_proto_goTypes, | ||||
| 		DependencyIndexes: file_asynq_proto_depIdxs, | ||||
| 		MessageInfos:      file_asynq_proto_msgTypes, | ||||
| 	}.Build() | ||||
| 	File_asynq_proto = out.File | ||||
| 	file_asynq_proto_rawDesc = nil | ||||
| 	file_asynq_proto_goTypes = nil | ||||
| 	file_asynq_proto_depIdxs = nil | ||||
| } | ||||
							
								
								
									
										163
									
								
								internal/proto/asynq.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								internal/proto/asynq.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| syntax = "proto3"; | ||||
| package asynq; | ||||
|  | ||||
| import "google/protobuf/timestamp.proto"; | ||||
|  | ||||
| option go_package = "github.com/hibiken/asynq/internal/proto"; | ||||
|  | ||||
| // TaskMessage is the internal representation of a task with additional | ||||
| // metadata fields. | ||||
| message TaskMessage { | ||||
| 	// Type indicates the kind of the task to be performed. | ||||
|   string type = 1; | ||||
|  | ||||
| 	// Payload holds data needed to process the task. | ||||
|   bytes payload = 2; | ||||
|  | ||||
| 	// Unique identifier for the task. | ||||
|   string id = 3; | ||||
|  | ||||
| 	// Name of the queue to which this task belongs. | ||||
|   string queue = 4; | ||||
|  | ||||
| 	// Max number of retries for this task. | ||||
|   int32 retry = 5; | ||||
|  | ||||
| 	// Number of times this task has been retried so far. | ||||
|   int32 retried = 6; | ||||
|  | ||||
| 	// Error message from the last failure. | ||||
|   string error_msg = 7; | ||||
|  | ||||
|   // Time of last failure in Unix time, | ||||
|   // the number of seconds elapsed since January 1, 1970 UTC. | ||||
|   // Use zero to indicate no last failure. | ||||
|   int64 last_failed_at = 11; | ||||
|  | ||||
| 	// Timeout specifies timeout in seconds. | ||||
| 	// Use zero to indicate no timeout. | ||||
|   int64 timeout = 8; | ||||
|  | ||||
| 	// Deadline specifies the deadline for the task in Unix time, | ||||
| 	// the number of seconds elapsed since January 1, 1970 UTC. | ||||
| 	// Use zero to indicate no deadline. | ||||
|   int64 deadline = 9; | ||||
|  | ||||
| 	// UniqueKey holds the redis key used for uniqueness lock for this task. | ||||
| 	// Empty string indicates that no uniqueness lock was used. | ||||
|   string unique_key = 10; | ||||
|  | ||||
|   // Retention period specified in a number of seconds. | ||||
|   // The task will be stored in redis as a completed task until the TTL | ||||
|   // expires. | ||||
|   int64 retention = 12; | ||||
|  | ||||
|   // Time when the task completed in success in Unix time, | ||||
|   // the number of seconds elapsed since January 1, 1970 UTC. | ||||
|   // This field is populated if result_ttl > 0 upon completion. | ||||
|   int64 completed_at = 13; | ||||
| }; | ||||
|  | ||||
| // ServerInfo holds information about a running server. | ||||
| message ServerInfo { | ||||
|   // Host machine the server is running on. | ||||
|   string host = 1; | ||||
|  | ||||
|   // PID of the server process. | ||||
|   int32 pid = 2; | ||||
|  | ||||
|   // Unique identifier for this server. | ||||
|   string server_id = 3; | ||||
|  | ||||
|   // Maximum number of concurrency this server will use. | ||||
|   int32 concurrency = 4; | ||||
|  | ||||
|   // List of queue names with their priorities. | ||||
|   // The server will consume tasks from the queues and prioritize | ||||
|   // queues with higher priority numbers. | ||||
|   map<string, int32> queues = 5; | ||||
|  | ||||
|   // If set, the server will always consume tasks from a queue with higher | ||||
|   // priority. | ||||
|   bool strict_priority = 6; | ||||
|  | ||||
|   // Status indicates the status of the server. | ||||
|   string status = 7; | ||||
|  | ||||
|   // Time this server was started. | ||||
|   google.protobuf.Timestamp start_time = 8; | ||||
|  | ||||
|   // Number of workers currently processing tasks. | ||||
|   int32 active_worker_count = 9; | ||||
| }; | ||||
|  | ||||
| // WorkerInfo holds information about a running worker. | ||||
| message WorkerInfo { | ||||
|   // Host matchine this worker is running on. | ||||
|   string host = 1; | ||||
|  | ||||
|   // PID of the process in which this worker is running. | ||||
|   int32 pid = 2; | ||||
|  | ||||
|   // ID of the server in which this worker is running. | ||||
|   string server_id = 3; | ||||
|  | ||||
|   // ID of the task this worker is processing. | ||||
|   string task_id = 4; | ||||
|  | ||||
|   // Type of the task this worker is processing. | ||||
|   string task_type = 5; | ||||
|  | ||||
|   // Payload of the task this worker is processing. | ||||
|   bytes task_payload = 6; | ||||
|  | ||||
|   // Name of the queue the task the worker is processing belongs. | ||||
|   string queue = 7; | ||||
|  | ||||
|   // Time this worker started processing the task. | ||||
|   google.protobuf.Timestamp start_time = 8; | ||||
|  | ||||
|   // Deadline by which the worker needs to complete processing  | ||||
|   // the task. If worker exceeds the deadline, the task will fail. | ||||
|   google.protobuf.Timestamp deadline = 9; | ||||
| }; | ||||
|  | ||||
| // SchedulerEntry holds information about a periodic task registered  | ||||
| // with a scheduler. | ||||
| message SchedulerEntry { | ||||
| 	// Identifier of the scheduler entry. | ||||
| 	string id = 1; | ||||
|  | ||||
| 	// Periodic schedule spec of the entry. | ||||
| 	string spec = 2; | ||||
|  | ||||
| 	// Task type of the periodic task. | ||||
| 	string task_type = 3; | ||||
|  | ||||
| 	// Task payload of the periodic task. | ||||
| 	bytes task_payload = 4; | ||||
|  | ||||
| 	// Options used to enqueue the periodic task. | ||||
| 	repeated string enqueue_options = 5; | ||||
|  | ||||
| 	// Next time the task will be enqueued. | ||||
|   google.protobuf.Timestamp next_enqueue_time = 6; | ||||
|  | ||||
| 	// Last time the task was enqueued. | ||||
| 	// Zero time if task was never enqueued. | ||||
|   google.protobuf.Timestamp prev_enqueue_time = 7; | ||||
| }; | ||||
|  | ||||
| // SchedulerEnqueueEvent holds information about an enqueue event | ||||
| // by a scheduler. | ||||
| message SchedulerEnqueueEvent { | ||||
| 	// ID of the task that was enqueued. | ||||
|   string task_id = 1; | ||||
|  | ||||
| 	// Time the task was enqueued. | ||||
|   google.protobuf.Timestamp enqueue_time = 2; | ||||
| }; | ||||
							
								
								
									
										266
									
								
								internal/rdb/benchmark_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										266
									
								
								internal/rdb/benchmark_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,266 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package rdb | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/asynqtest" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| func BenchmarkEnqueue(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	msg := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Enqueue(msg); err != nil { | ||||
| 			b.Fatalf("Enqueue failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkEnqueueUnique(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	msg := &base.TaskMessage{ | ||||
| 		Type:      "task1", | ||||
| 		Payload:   nil, | ||||
| 		Queue:     base.DefaultQueueName, | ||||
| 		UniqueKey: base.UniqueKey("default", "task1", nil), | ||||
| 	} | ||||
| 	uniqueTTL := 5 * time.Minute | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.EnqueueUnique(msg, uniqueTTL); err != nil { | ||||
| 			b.Fatalf("EnqueueUnique failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkSchedule(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	msg := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	processAt := time.Now().Add(3 * time.Minute) | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Schedule(msg, processAt); err != nil { | ||||
| 			b.Fatalf("Schedule failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkScheduleUnique(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	msg := &base.TaskMessage{ | ||||
| 		Type:      "task1", | ||||
| 		Payload:   nil, | ||||
| 		Queue:     base.DefaultQueueName, | ||||
| 		UniqueKey: base.UniqueKey("default", "task1", nil), | ||||
| 	} | ||||
| 	processAt := time.Now().Add(3 * time.Minute) | ||||
| 	uniqueTTL := 5 * time.Minute | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.ScheduleUnique(msg, processAt, uniqueTTL); err != nil { | ||||
| 			b.Fatalf("EnqueueUnique failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkDequeueSingleQueue(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		for i := 0; i < 10; i++ { | ||||
| 			m := asynqtest.NewTaskMessageWithQueue( | ||||
| 				fmt.Sprintf("task%d", i), nil, base.DefaultQueueName) | ||||
| 			if err := r.Enqueue(m); err != nil { | ||||
| 				b.Fatalf("Enqueue failed: %v", err) | ||||
| 			} | ||||
| 		} | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if _, _, err := r.Dequeue(base.DefaultQueueName); err != nil { | ||||
| 			b.Fatalf("Dequeue failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkDequeueMultipleQueues(b *testing.B) { | ||||
| 	qnames := []string{"critical", "default", "low"} | ||||
| 	r := setup(b) | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		for i := 0; i < 10; i++ { | ||||
| 			for _, qname := range qnames { | ||||
| 				m := asynqtest.NewTaskMessageWithQueue( | ||||
| 					fmt.Sprintf("%s_task%d", qname, i), nil, qname) | ||||
| 				if err := r.Enqueue(m); err != nil { | ||||
| 					b.Fatalf("Enqueue failed: %v", err) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if _, _, err := r.Dequeue(qnames...); err != nil { | ||||
| 			b.Fatalf("Dequeue failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkDone(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	m1 := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	m2 := asynqtest.NewTaskMessage("task2", nil) | ||||
| 	m3 := asynqtest.NewTaskMessage("task3", nil) | ||||
| 	msgs := []*base.TaskMessage{m1, m2, m3} | ||||
| 	zs := []base.Z{ | ||||
| 		{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()}, | ||||
| 		{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()}, | ||||
| 		{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()}, | ||||
| 	} | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName) | ||||
| 		asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Done(msgs[0]); err != nil { | ||||
| 			b.Fatalf("Done failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkRetry(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	m1 := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	m2 := asynqtest.NewTaskMessage("task2", nil) | ||||
| 	m3 := asynqtest.NewTaskMessage("task3", nil) | ||||
| 	msgs := []*base.TaskMessage{m1, m2, m3} | ||||
| 	zs := []base.Z{ | ||||
| 		{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()}, | ||||
| 		{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()}, | ||||
| 		{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()}, | ||||
| 	} | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName) | ||||
| 		asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Retry(msgs[0], time.Now().Add(1*time.Minute), "error", true /*isFailure*/); err != nil { | ||||
| 			b.Fatalf("Retry failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkArchive(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	m1 := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	m2 := asynqtest.NewTaskMessage("task2", nil) | ||||
| 	m3 := asynqtest.NewTaskMessage("task3", nil) | ||||
| 	msgs := []*base.TaskMessage{m1, m2, m3} | ||||
| 	zs := []base.Z{ | ||||
| 		{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()}, | ||||
| 		{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()}, | ||||
| 		{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()}, | ||||
| 	} | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName) | ||||
| 		asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Archive(msgs[0], "error"); err != nil { | ||||
| 			b.Fatalf("Archive failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkRequeue(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	m1 := asynqtest.NewTaskMessage("task1", nil) | ||||
| 	m2 := asynqtest.NewTaskMessage("task2", nil) | ||||
| 	m3 := asynqtest.NewTaskMessage("task3", nil) | ||||
| 	msgs := []*base.TaskMessage{m1, m2, m3} | ||||
| 	zs := []base.Z{ | ||||
| 		{Message: m1, Score: time.Now().Add(10 * time.Second).Unix()}, | ||||
| 		{Message: m2, Score: time.Now().Add(20 * time.Second).Unix()}, | ||||
| 		{Message: m3, Score: time.Now().Add(30 * time.Second).Unix()}, | ||||
| 	} | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		asynqtest.SeedActiveQueue(b, r.client, msgs, base.DefaultQueueName) | ||||
| 		asynqtest.SeedDeadlines(b, r.client, zs, base.DefaultQueueName) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.Requeue(msgs[0]); err != nil { | ||||
| 			b.Fatalf("Requeue failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkCheckAndEnqueue(b *testing.B) { | ||||
| 	r := setup(b) | ||||
| 	now := time.Now() | ||||
| 	var zs []base.Z | ||||
| 	for i := -100; i < 100; i++ { | ||||
| 		msg := asynqtest.NewTaskMessage(fmt.Sprintf("task%d", i), nil) | ||||
| 		score := now.Add(time.Duration(i) * time.Second).Unix() | ||||
| 		zs = append(zs, base.Z{Message: msg, Score: score}) | ||||
| 	} | ||||
| 	b.ResetTimer() | ||||
|  | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		b.StopTimer() | ||||
| 		asynqtest.FlushDB(b, r.client) | ||||
| 		asynqtest.SeedScheduledQueue(b, r.client, zs, base.DefaultQueueName) | ||||
| 		b.StartTimer() | ||||
|  | ||||
| 		if err := r.ForwardIfReady(base.DefaultQueueName); err != nil { | ||||
| 			b.Fatalf("ForwardIfReady failed: %v", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -10,7 +10,7 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| @@ -81,6 +81,15 @@ func (tb *TestBroker) Done(msg *base.TaskMessage) error { | ||||
| 	return tb.real.Done(msg) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) MarkAsComplete(msg *base.TaskMessage) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return errRedisDown | ||||
| 	} | ||||
| 	return tb.real.MarkAsComplete(msg) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) Requeue(msg *base.TaskMessage) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| @@ -108,13 +117,13 @@ func (tb *TestBroker) ScheduleUnique(msg *base.TaskMessage, processAt time.Time, | ||||
| 	return tb.real.ScheduleUnique(msg, processAt, ttl) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) Retry(msg *base.TaskMessage, processAt time.Time, errMsg string) error { | ||||
| func (tb *TestBroker) Retry(msg *base.TaskMessage, processAt time.Time, errMsg string, isFailure bool) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return errRedisDown | ||||
| 	} | ||||
| 	return tb.real.Retry(msg, processAt, errMsg) | ||||
| 	return tb.real.Retry(msg, processAt, errMsg, isFailure) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) Archive(msg *base.TaskMessage, errMsg string) error { | ||||
| @@ -126,13 +135,22 @@ func (tb *TestBroker) Archive(msg *base.TaskMessage, errMsg string) error { | ||||
| 	return tb.real.Archive(msg, errMsg) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) CheckAndEnqueue(qnames ...string) error { | ||||
| func (tb *TestBroker) ForwardIfReady(qnames ...string) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return errRedisDown | ||||
| 	} | ||||
| 	return tb.real.CheckAndEnqueue(qnames...) | ||||
| 	return tb.real.ForwardIfReady(qnames...) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) DeleteExpiredCompletedTasks(qname string) error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return errRedisDown | ||||
| 	} | ||||
| 	return tb.real.DeleteExpiredCompletedTasks(qname) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) ListDeadlineExceeded(deadline time.Time, qnames ...string) ([]*base.TaskMessage, error) { | ||||
| @@ -180,6 +198,15 @@ func (tb *TestBroker) PublishCancelation(id string) error { | ||||
| 	return tb.real.PublishCancelation(id) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) WriteResult(qname, id string, data []byte) (int, error) { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
| 	if tb.sleeping { | ||||
| 		return 0, errRedisDown | ||||
| 	} | ||||
| 	return tb.real.WriteResult(qname, id, data) | ||||
| } | ||||
|  | ||||
| func (tb *TestBroker) Ping() error { | ||||
| 	tb.mu.Lock() | ||||
| 	defer tb.mu.Unlock() | ||||
|   | ||||
							
								
								
									
										81
									
								
								janitor.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								janitor.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| // Copyright 2021 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| ) | ||||
|  | ||||
| // A janitor is responsible for deleting expired completed tasks from the specified | ||||
| // queues. It periodically checks for any expired tasks in the completed set, and | ||||
| // deletes them. | ||||
| type janitor struct { | ||||
| 	logger *log.Logger | ||||
| 	broker base.Broker | ||||
|  | ||||
| 	// channel to communicate back to the long running "janitor" goroutine. | ||||
| 	done chan struct{} | ||||
|  | ||||
| 	// list of queue names to check. | ||||
| 	queues []string | ||||
|  | ||||
| 	// average interval between checks. | ||||
| 	avgInterval time.Duration | ||||
| } | ||||
|  | ||||
| type janitorParams struct { | ||||
| 	logger   *log.Logger | ||||
| 	broker   base.Broker | ||||
| 	queues   []string | ||||
| 	interval time.Duration | ||||
| } | ||||
|  | ||||
| func newJanitor(params janitorParams) *janitor { | ||||
| 	return &janitor{ | ||||
| 		logger:      params.logger, | ||||
| 		broker:      params.broker, | ||||
| 		done:        make(chan struct{}), | ||||
| 		queues:      params.queues, | ||||
| 		avgInterval: params.interval, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (j *janitor) shutdown() { | ||||
| 	j.logger.Debug("Janitor shutting down...") | ||||
| 	// Signal the janitor goroutine to stop. | ||||
| 	j.done <- struct{}{} | ||||
| } | ||||
|  | ||||
| // start starts the "janitor" goroutine. | ||||
| func (j *janitor) start(wg *sync.WaitGroup) { | ||||
| 	wg.Add(1) | ||||
| 	timer := time.NewTimer(j.avgInterval) // randomize this interval with margin of 1s | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-j.done: | ||||
| 				j.logger.Debug("Janitor done") | ||||
| 				return | ||||
| 			case <-timer.C: | ||||
| 				j.exec() | ||||
| 				timer.Reset(j.avgInterval) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| } | ||||
|  | ||||
| func (j *janitor) exec() { | ||||
| 	for _, qname := range j.queues { | ||||
| 		if err := j.broker.DeleteExpiredCompletedTasks(qname); err != nil { | ||||
| 			j.logger.Errorf("Could not delete expired completed tasks from queue %q: %v", | ||||
| 				qname, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										89
									
								
								janitor_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								janitor_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | ||||
| // Copyright 2021 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	h "github.com/hibiken/asynq/internal/asynqtest" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| ) | ||||
|  | ||||
| func newCompletedTask(qname, tasktype string, payload []byte, completedAt time.Time) *base.TaskMessage { | ||||
| 	msg := h.NewTaskMessageWithQueue(tasktype, payload, qname) | ||||
| 	msg.CompletedAt = completedAt.Unix() | ||||
| 	return msg | ||||
| } | ||||
|  | ||||
| func TestJanitor(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
| 	const interval = 1 * time.Second | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:   testLogger, | ||||
| 		broker:   rdbClient, | ||||
| 		queues:   []string{"default", "custom"}, | ||||
| 		interval: interval, | ||||
| 	}) | ||||
|  | ||||
| 	now := time.Now() | ||||
| 	hourAgo := now.Add(-1 * time.Hour) | ||||
| 	minuteAgo := now.Add(-1 * time.Minute) | ||||
| 	halfHourAgo := now.Add(-30 * time.Minute) | ||||
| 	halfHourFromNow := now.Add(30 * time.Minute) | ||||
| 	fiveMinFromNow := now.Add(5 * time.Minute) | ||||
| 	msg1 := newCompletedTask("default", "task1", nil, hourAgo) | ||||
| 	msg2 := newCompletedTask("default", "task2", nil, minuteAgo) | ||||
| 	msg3 := newCompletedTask("custom", "task3", nil, hourAgo) | ||||
| 	msg4 := newCompletedTask("custom", "task4", nil, minuteAgo) | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		completed     map[string][]base.Z // initial completed sets | ||||
| 		wantCompleted map[string][]base.Z // expected completed sets after janitor runs | ||||
| 	}{ | ||||
| 		{ | ||||
| 			completed: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: msg1, Score: halfHourAgo.Unix()}, | ||||
| 					{Message: msg2, Score: fiveMinFromNow.Unix()}, | ||||
| 				}, | ||||
| 				"custom": { | ||||
| 					{Message: msg3, Score: halfHourFromNow.Unix()}, | ||||
| 					{Message: msg4, Score: minuteAgo.Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wantCompleted: map[string][]base.Z{ | ||||
| 				"default": { | ||||
| 					{Message: msg2, Score: fiveMinFromNow.Unix()}, | ||||
| 				}, | ||||
| 				"custom": { | ||||
| 					{Message: msg3, Score: halfHourFromNow.Unix()}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r) | ||||
| 		h.SeedAllCompletedQueues(t, r, tc.completed) | ||||
|  | ||||
| 		var wg sync.WaitGroup | ||||
| 		janitor.start(&wg) | ||||
| 		time.Sleep(2 * interval) // make sure to let janitor run at least one time | ||||
| 		janitor.shutdown() | ||||
|  | ||||
| 		for qname, want := range tc.wantCompleted { | ||||
| 			got := h.GetCompletedEntries(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, got, h.SortZSetEntryOpt); diff != "" { | ||||
| 				t.Errorf("diff found in %q after running janitor: (-want, +got)\n%s", base.CompletedKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										230
									
								
								payload.go
									
									
									
									
									
								
							
							
						
						
									
										230
									
								
								payload.go
									
									
									
									
									
								
							| @@ -1,230 +0,0 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/spf13/cast" | ||||
| ) | ||||
|  | ||||
| // Payload holds arbitrary data needed for task execution. | ||||
| type Payload struct { | ||||
| 	data map[string]interface{} | ||||
| } | ||||
|  | ||||
| type errKeyNotFound struct { | ||||
| 	key string | ||||
| } | ||||
|  | ||||
| func (e *errKeyNotFound) Error() string { | ||||
| 	return fmt.Sprintf("key %q does not exist", e.key) | ||||
| } | ||||
|  | ||||
| // Has reports whether key exists. | ||||
| func (p Payload) Has(key string) bool { | ||||
| 	_, ok := p.data[key] | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| func toInt(v interface{}) (int, error) { | ||||
| 	switch v := v.(type) { | ||||
| 	case json.Number: | ||||
| 		val, err := v.Int64() | ||||
| 		if err != nil { | ||||
| 			return 0, err | ||||
| 		} | ||||
| 		return int(val), nil | ||||
| 	default: | ||||
| 		return cast.ToIntE(v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // String returns a string representation of payload data. | ||||
| func (p Payload) String() string { | ||||
| 	return fmt.Sprint(p.data) | ||||
| } | ||||
|  | ||||
| // MarshalJSON returns the JSON encoding of payload data. | ||||
| func (p Payload) MarshalJSON() ([]byte, error) { | ||||
| 	return json.Marshal(p.data) | ||||
| } | ||||
|  | ||||
| // GetString returns a string value if a string type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetString(key string) (string, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return "", &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringE(v) | ||||
| } | ||||
|  | ||||
| // GetInt returns an int value if a numeric type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetInt(key string) (int, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return 0, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return toInt(v) | ||||
| } | ||||
|  | ||||
| // GetFloat64 returns a float64 value if a numeric type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetFloat64(key string) (float64, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return 0, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	switch v := v.(type) { | ||||
| 	case json.Number: | ||||
| 		return v.Float64() | ||||
| 	default: | ||||
| 		return cast.ToFloat64E(v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // GetBool returns a boolean value if a boolean type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetBool(key string) (bool, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return false, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToBoolE(v) | ||||
| } | ||||
|  | ||||
| // GetStringSlice returns a slice of strings if a string slice type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetStringSlice(key string) ([]string, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringSliceE(v) | ||||
| } | ||||
|  | ||||
| // GetIntSlice returns a slice of ints if a int slice type is associated with | ||||
| // the key, otherwise reports an error. | ||||
| func (p Payload) GetIntSlice(key string) ([]int, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	switch v := v.(type) { | ||||
| 	case []interface{}: | ||||
| 		var res []int | ||||
| 		for _, elem := range v { | ||||
| 			val, err := toInt(elem) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			res = append(res, int(val)) | ||||
| 		} | ||||
| 		return res, nil | ||||
| 	default: | ||||
| 		return cast.ToIntSliceE(v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // GetStringMap returns a map of string to empty interface | ||||
| // if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetStringMap(key string) (map[string]interface{}, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringMapE(v) | ||||
| } | ||||
|  | ||||
| // GetStringMapString returns a map of string to string | ||||
| // if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetStringMapString(key string) (map[string]string, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringMapStringE(v) | ||||
| } | ||||
|  | ||||
| // GetStringMapStringSlice returns a map of string to string slice | ||||
| // if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetStringMapStringSlice(key string) (map[string][]string, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringMapStringSliceE(v) | ||||
| } | ||||
|  | ||||
| // GetStringMapInt returns a map of string to int | ||||
| // if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetStringMapInt(key string) (map[string]int, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	switch v := v.(type) { | ||||
| 	case map[string]interface{}: | ||||
| 		res := make(map[string]int) | ||||
| 		for key, val := range v { | ||||
| 			ival, err := toInt(val) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 			res[key] = ival | ||||
| 		} | ||||
| 		return res, nil | ||||
| 	default: | ||||
| 		return cast.ToStringMapIntE(v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // GetStringMapBool returns a map of string to boolean | ||||
| // if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetStringMapBool(key string) (map[string]bool, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return nil, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToStringMapBoolE(v) | ||||
| } | ||||
|  | ||||
| // GetTime returns a time value if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetTime(key string) (time.Time, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return time.Time{}, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	return cast.ToTimeE(v) | ||||
| } | ||||
|  | ||||
| // GetDuration returns a duration value if a correct map type is associated with the key, | ||||
| // otherwise reports an error. | ||||
| func (p Payload) GetDuration(key string) (time.Duration, error) { | ||||
| 	v, ok := p.data[key] | ||||
| 	if !ok { | ||||
| 		return 0, &errKeyNotFound{key} | ||||
| 	} | ||||
| 	switch v := v.(type) { | ||||
| 	case json.Number: | ||||
| 		val, err := v.Int64() | ||||
| 		if err != nil { | ||||
| 			return 0, err | ||||
| 		} | ||||
| 		return time.Duration(val), nil | ||||
| 	default: | ||||
| 		return cast.ToDurationE(v) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										675
									
								
								payload_test.go
									
									
									
									
									
								
							
							
						
						
									
										675
									
								
								payload_test.go
									
									
									
									
									
								
							| @@ -1,675 +0,0 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/go-cmp/cmp" | ||||
| 	"github.com/google/go-cmp/cmp/cmpopts" | ||||
| 	h "github.com/hibiken/asynq/internal/asynqtest" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| type payloadTest struct { | ||||
| 	data   map[string]interface{} | ||||
| 	key    string | ||||
| 	nonkey string | ||||
| } | ||||
|  | ||||
| func TestPayloadString(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"name": "gopher"}, | ||||
| 			key:    "name", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetString(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("Payload.GetString(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetString(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("With Marshaling: Payload.GetString(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetString(tc.nonkey) | ||||
| 		if err == nil || got != "" { | ||||
| 			t.Errorf("Payload.GetString(%q) = %v, %v; want '', error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadInt(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"user_id": 42}, | ||||
| 			key:    "user_id", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetInt(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("Payload.GetInt(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetInt(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("With Marshaling: Payload.GetInt(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetInt(tc.nonkey) | ||||
| 		if err == nil || got != 0 { | ||||
| 			t.Errorf("Payload.GetInt(%q) = %v, %v; want 0, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadFloat64(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"pi": 3.14}, | ||||
| 			key:    "pi", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetFloat64(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("Payload.GetFloat64(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetFloat64(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("With Marshaling: Payload.GetFloat64(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetFloat64(tc.nonkey) | ||||
| 		if err == nil || got != 0 { | ||||
| 			t.Errorf("Payload.GetFloat64(%q) = %v, %v; want 0, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadBool(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"enabled": true}, | ||||
| 			key:    "enabled", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetBool(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("Payload.GetBool(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetBool(tc.key) | ||||
| 		if err != nil || got != tc.data[tc.key] { | ||||
| 			t.Errorf("With Marshaling: Payload.GetBool(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetBool(tc.nonkey) | ||||
| 		if err == nil || got != false { | ||||
| 			t.Errorf("Payload.GetBool(%q) = %v, %v; want false, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringSlice(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"names": []string{"luke", "rey", "anakin"}}, | ||||
| 			key:    "names", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringSlice(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringSlice(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringSlice(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringSlice(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadIntSlice(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"nums": []int{9, 8, 7}}, | ||||
| 			key:    "nums", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetIntSlice(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetIntSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetIntSlice(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetIntSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetIntSlice(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetIntSlice(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringMap(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"user": map[string]interface{}{"name": "Jon Doe", "score": 2.2}}, | ||||
| 			key:    "user", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringMap(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringMap(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringMap(tc.key) | ||||
| 		ignoreOpt := cmpopts.IgnoreMapEntries(func(key string, val interface{}) bool { | ||||
| 			switch val.(type) { | ||||
| 			case json.Number: | ||||
| 				return true | ||||
| 			default: | ||||
| 				return false | ||||
| 			} | ||||
| 		}) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key], ignoreOpt) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringMap(%q) = %v, %v, want %v, nil;(-want,+got)\n%s", | ||||
| 				tc.key, got, err, tc.data[tc.key], diff) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringMap(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringMap(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringMapString(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"address": map[string]string{"line": "123 Main St", "city": "San Francisco", "state": "CA"}}, | ||||
| 			key:    "address", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringMapString(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringMapString(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringMapString(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringMapString(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringMapString(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringMapString(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringMapStringSlice(t *testing.T) { | ||||
| 	favs := map[string][]string{ | ||||
| 		"movies":   {"forrest gump", "star wars"}, | ||||
| 		"tv_shows": {"game of thrones", "HIMYM", "breaking bad"}, | ||||
| 	} | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"favorites": favs}, | ||||
| 			key:    "favorites", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringMapStringSlice(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringMapStringSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringMapStringSlice(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringMapStringSlice(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringMapStringSlice(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringMapStringSlice(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringMapInt(t *testing.T) { | ||||
| 	counter := map[string]int{ | ||||
| 		"a": 1, | ||||
| 		"b": 101, | ||||
| 		"c": 42, | ||||
| 	} | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"counts": counter}, | ||||
| 			key:    "counts", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringMapInt(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringMapInt(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringMapInt(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringMapInt(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringMapInt(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringMapInt(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadStringMapBool(t *testing.T) { | ||||
| 	features := map[string]bool{ | ||||
| 		"A": false, | ||||
| 		"B": true, | ||||
| 		"C": true, | ||||
| 	} | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"features": features}, | ||||
| 			key:    "features", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetStringMapBool(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetStringMapBool(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetStringMapBool(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetStringMapBool(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetStringMapBool(tc.nonkey) | ||||
| 		if err == nil || got != nil { | ||||
| 			t.Errorf("Payload.GetStringMapBool(%q) = %v, %v; want nil, error", | ||||
| 				tc.key, got, err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadTime(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"current": time.Now()}, | ||||
| 			key:    "current", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetTime(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetTime(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetTime(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetTime(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetTime(tc.nonkey) | ||||
| 		if err == nil || !got.IsZero() { | ||||
| 			t.Errorf("Payload.GetTime(%q) = %v, %v; want %v, error", | ||||
| 				tc.key, got, err, time.Time{}) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadDuration(t *testing.T) { | ||||
| 	tests := []payloadTest{ | ||||
| 		{ | ||||
| 			data:   map[string]interface{}{"duration": 15 * time.Minute}, | ||||
| 			key:    "duration", | ||||
| 			nonkey: "unknown", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		payload := Payload{tc.data} | ||||
|  | ||||
| 		got, err := payload.GetDuration(tc.key) | ||||
| 		diff := cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("Payload.GetDuration(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// encode and then decode task messsage. | ||||
| 		in := h.NewTaskMessage("testing", tc.data) | ||||
| 		encoded, err := base.EncodeMessage(in) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		out, err := base.DecodeMessage(encoded) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		payload = Payload{out.Payload} | ||||
| 		got, err = payload.GetDuration(tc.key) | ||||
| 		diff = cmp.Diff(got, tc.data[tc.key]) | ||||
| 		if err != nil || diff != "" { | ||||
| 			t.Errorf("With Marshaling: Payload.GetDuration(%q) = %v, %v, want %v, nil", | ||||
| 				tc.key, got, err, tc.data[tc.key]) | ||||
| 		} | ||||
|  | ||||
| 		// access non-existent key. | ||||
| 		got, err = payload.GetDuration(tc.nonkey) | ||||
| 		if err == nil || got != 0 { | ||||
| 			t.Errorf("Payload.GetDuration(%q) = %v, %v; want %v, error", | ||||
| 				tc.key, got, err, time.Duration(0)) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadHas(t *testing.T) { | ||||
| 	payload := Payload{map[string]interface{}{ | ||||
| 		"user_id": 123, | ||||
| 	}} | ||||
|  | ||||
| 	if !payload.Has("user_id") { | ||||
| 		t.Errorf("Payload.Has(%q) = false, want true", "user_id") | ||||
| 	} | ||||
| 	if payload.Has("name") { | ||||
| 		t.Errorf("Payload.Has(%q) = true, want false", "name") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPayloadDebuggingStrings(t *testing.T) { | ||||
| 	data := map[string]interface{}{ | ||||
| 		"foo": 123, | ||||
| 		"bar": "hello", | ||||
| 		"baz": false, | ||||
| 	} | ||||
| 	payload := Payload{data: data} | ||||
|  | ||||
| 	if payload.String() != fmt.Sprint(data) { | ||||
| 		t.Errorf("Payload.String() = %q, want %q", | ||||
| 			payload.String(), fmt.Sprint(data)) | ||||
| 	} | ||||
|  | ||||
| 	got, err := payload.MarshalJSON() | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	want, err := json.Marshal(data) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if diff := cmp.Diff(got, want); diff != "" { | ||||
| 		t.Errorf("Payload.MarhsalJSON() = %s, want %s; (-want,+got)\n%s", | ||||
| 			got, want, diff) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										92
									
								
								processor.go
									
									
									
									
									
								
							
							
						
						
									
										92
									
								
								processor.go
									
									
									
									
									
								
							| @@ -6,7 +6,6 @@ package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math/rand" | ||||
| 	"runtime" | ||||
| @@ -17,8 +16,9 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"golang.org/x/time/rate" | ||||
| ) | ||||
|  | ||||
| @@ -34,6 +34,7 @@ type processor struct { | ||||
| 	orderedQueues []string | ||||
|  | ||||
| 	retryDelayFunc RetryDelayFunc | ||||
| 	isFailureFunc  func(error) bool | ||||
|  | ||||
| 	errHandler ErrorHandler | ||||
|  | ||||
| @@ -63,7 +64,7 @@ type processor struct { | ||||
| 	// cancelations is a set of cancel functions for all active tasks. | ||||
| 	cancelations *base.Cancelations | ||||
|  | ||||
| 	starting chan<- *base.TaskMessage | ||||
| 	starting chan<- *workerInfo | ||||
| 	finished chan<- *base.TaskMessage | ||||
| } | ||||
|  | ||||
| @@ -71,6 +72,7 @@ type processorParams struct { | ||||
| 	logger          *log.Logger | ||||
| 	broker          base.Broker | ||||
| 	retryDelayFunc  RetryDelayFunc | ||||
| 	isFailureFunc   func(error) bool | ||||
| 	syncCh          chan<- *syncRequest | ||||
| 	cancelations    *base.Cancelations | ||||
| 	concurrency     int | ||||
| @@ -78,7 +80,7 @@ type processorParams struct { | ||||
| 	strictPriority  bool | ||||
| 	errHandler      ErrorHandler | ||||
| 	shutdownTimeout time.Duration | ||||
| 	starting        chan<- *base.TaskMessage | ||||
| 	starting        chan<- *workerInfo | ||||
| 	finished        chan<- *base.TaskMessage | ||||
| } | ||||
|  | ||||
| @@ -95,6 +97,7 @@ func newProcessor(params processorParams) *processor { | ||||
| 		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), | ||||
| @@ -123,8 +126,8 @@ func (p *processor) stop() { | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| // NOTE: once terminated, processor cannot be re-started. | ||||
| func (p *processor) terminate() { | ||||
| // NOTE: once shutdown, processor cannot be re-started. | ||||
| func (p *processor) shutdown() { | ||||
| 	p.stop() | ||||
|  | ||||
| 	time.AfterFunc(p.shutdownTimeout, func() { close(p.abort) }) | ||||
| @@ -163,7 +166,7 @@ func (p *processor) exec() { | ||||
| 		qnames := p.queues() | ||||
| 		msg, deadline, err := p.broker.Dequeue(qnames...) | ||||
| 		switch { | ||||
| 		case err == rdb.ErrNoProcessableTask: | ||||
| 		case errors.Is(err, errors.ErrNoProcessableTask): | ||||
| 			p.logger.Debug("All queues are empty") | ||||
| 			// Queues are empty, this is a normal behavior. | ||||
| 			// Sleep to avoid slamming redis and let scheduler move tasks into queues. | ||||
| @@ -180,32 +183,42 @@ func (p *processor) exec() { | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		p.starting <- msg | ||||
| 		p.starting <- &workerInfo{msg, time.Now(), deadline} | ||||
| 		go func() { | ||||
| 			defer func() { | ||||
| 				p.finished <- msg | ||||
| 				<-p.sema // release token | ||||
| 			}() | ||||
|  | ||||
| 			ctx, cancel := createContext(msg, deadline) | ||||
| 			p.cancelations.Add(msg.ID.String(), cancel) | ||||
| 			ctx, cancel := asynqcontext.New(msg, deadline) | ||||
| 			p.cancelations.Add(msg.ID, cancel) | ||||
| 			defer func() { | ||||
| 				cancel() | ||||
| 				p.cancelations.Delete(msg.ID.String()) | ||||
| 				p.cancelations.Delete(msg.ID) | ||||
| 			}() | ||||
|  | ||||
| 			// check context before starting a worker goroutine. | ||||
| 			select { | ||||
| 			case <-ctx.Done(): | ||||
| 				// already canceled (e.g. deadline exceeded). | ||||
| 				p.retryOrKill(ctx, msg, ctx.Err()) | ||||
| 				p.handleFailedMessage(ctx, msg, ctx.Err()) | ||||
| 				return | ||||
| 			default: | ||||
| 			} | ||||
|  | ||||
| 			resCh := make(chan error, 1) | ||||
| 			go func() { | ||||
| 				resCh <- p.perform(ctx, NewTask(msg.Type, msg.Payload)) | ||||
| 				task := newTask( | ||||
| 					msg.Type, | ||||
| 					msg.Payload, | ||||
| 					&ResultWriter{ | ||||
| 						id:     msg.ID, | ||||
| 						qname:  msg.Queue, | ||||
| 						broker: p.broker, | ||||
| 						ctx:    ctx, | ||||
| 					}, | ||||
| 				) | ||||
| 				resCh <- p.perform(ctx, task) | ||||
| 			}() | ||||
|  | ||||
| 			select { | ||||
| @@ -215,18 +228,14 @@ func (p *processor) exec() { | ||||
| 				p.requeue(msg) | ||||
| 				return | ||||
| 			case <-ctx.Done(): | ||||
| 				p.retryOrKill(ctx, msg, ctx.Err()) | ||||
| 				p.handleFailedMessage(ctx, msg, ctx.Err()) | ||||
| 				return | ||||
| 			case resErr := <-resCh: | ||||
| 				// Note: One of three things should happen. | ||||
| 				// 1) Done     -> Removes the message from Active | ||||
| 				// 2) Retry    -> Removes the message from Active & Adds the message to Retry | ||||
| 				// 3) Archive  -> Removes the message from Active & Adds the message to archive | ||||
| 				if resErr != nil { | ||||
| 					p.retryOrKill(ctx, msg, resErr) | ||||
| 					p.handleFailedMessage(ctx, msg, resErr) | ||||
| 					return | ||||
| 				} | ||||
| 				p.markAsDone(ctx, msg) | ||||
| 				p.handleSucceededMessage(ctx, msg) | ||||
| 			} | ||||
| 		}() | ||||
| 	} | ||||
| @@ -241,6 +250,34 @@ func (p *processor) requeue(msg *base.TaskMessage) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *processor) handleSucceededMessage(ctx context.Context, msg *base.TaskMessage) { | ||||
| 	if msg.Retention > 0 { | ||||
| 		p.markAsComplete(ctx, msg) | ||||
| 	} else { | ||||
| 		p.markAsDone(ctx, msg) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *processor) markAsComplete(ctx context.Context, msg *base.TaskMessage) { | ||||
| 	err := p.broker.MarkAsComplete(msg) | ||||
| 	if err != nil { | ||||
| 		errMsg := fmt.Sprintf("Could not move task id=%s type=%q from %q to %q:  %+v", | ||||
| 			msg.ID, msg.Type, base.ActiveKey(msg.Queue), base.CompletedKey(msg.Queue), err) | ||||
| 		deadline, ok := ctx.Deadline() | ||||
| 		if !ok { | ||||
| 			panic("asynq: internal error: missing deadline in context") | ||||
| 		} | ||||
| 		p.logger.Warnf("%s; Will retry syncing", errMsg) | ||||
| 		p.syncRequestCh <- &syncRequest{ | ||||
| 			fn: func() error { | ||||
| 				return p.broker.MarkAsComplete(msg) | ||||
| 			}, | ||||
| 			errMsg:   errMsg, | ||||
| 			deadline: deadline, | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *processor) markAsDone(ctx context.Context, msg *base.TaskMessage) { | ||||
| 	err := p.broker.Done(msg) | ||||
| 	if err != nil { | ||||
| @@ -264,22 +301,27 @@ func (p *processor) markAsDone(ctx context.Context, msg *base.TaskMessage) { | ||||
| // the task should not be retried and should be archived instead. | ||||
| var SkipRetry = errors.New("skip retry for the task") | ||||
|  | ||||
| func (p *processor) retryOrKill(ctx context.Context, msg *base.TaskMessage, err error) { | ||||
| func (p *processor) handleFailedMessage(ctx context.Context, 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(ctx, msg, err, false /*isFailure*/) | ||||
| 		return | ||||
| 	} | ||||
| 	if msg.Retried >= msg.Retry || errors.Is(err, SkipRetry) { | ||||
| 		p.logger.Warnf("Retry exhausted for task id=%s", msg.ID) | ||||
| 		p.archive(ctx, msg, err) | ||||
| 	} else { | ||||
| 		p.retry(ctx, msg, err) | ||||
| 		p.retry(ctx, msg, err, true /*isFailure*/) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (p *processor) retry(ctx context.Context, msg *base.TaskMessage, e error) { | ||||
| func (p *processor) retry(ctx context.Context, msg *base.TaskMessage, e error, isFailure bool) { | ||||
| 	d := p.retryDelayFunc(msg.Retried, e, NewTask(msg.Type, msg.Payload)) | ||||
| 	retryAt := time.Now().Add(d) | ||||
| 	err := p.broker.Retry(msg, retryAt, e.Error()) | ||||
| 	err := p.broker.Retry(msg, retryAt, e.Error(), isFailure) | ||||
| 	if err != nil { | ||||
| 		errMsg := fmt.Sprintf("Could not move task id=%s from %q to %q", msg.ID, base.ActiveKey(msg.Queue), base.RetryKey(msg.Queue)) | ||||
| 		deadline, ok := ctx.Deadline() | ||||
| @@ -289,7 +331,7 @@ func (p *processor) retry(ctx context.Context, msg *base.TaskMessage, e error) { | ||||
| 		p.logger.Warnf("%s; Will retry syncing", errMsg) | ||||
| 		p.syncRequestCh <- &syncRequest{ | ||||
| 			fn: func() error { | ||||
| 				return p.broker.Retry(msg, retryAt, e.Error()) | ||||
| 				return p.broker.Retry(msg, retryAt, e.Error(), isFailure) | ||||
| 			}, | ||||
| 			errMsg:   errMsg, | ||||
| 			deadline: deadline, | ||||
|   | ||||
| @@ -6,6 +6,7 @@ package asynq | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"sync" | ||||
| @@ -19,8 +20,14 @@ import ( | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| ) | ||||
|  | ||||
| var taskCmpOpts = []cmp.Option{ | ||||
| 	sortTaskOpt,                               // sort the tasks | ||||
| 	cmp.AllowUnexported(Task{}),               // allow typename, payload fields to be compared | ||||
| 	cmpopts.IgnoreFields(Task{}, "opts", "w"), // ignore opts, w fields | ||||
| } | ||||
|  | ||||
| // fakeHeartbeater receives from starting and finished channels and do nothing. | ||||
| func fakeHeartbeater(starting, finished <-chan *base.TaskMessage, done <-chan struct{}) { | ||||
| func fakeHeartbeater(starting <-chan *workerInfo, finished <-chan *base.TaskMessage, done <-chan struct{}) { | ||||
| 	for { | ||||
| 		select { | ||||
| 		case <-starting: | ||||
| @@ -42,8 +49,37 @@ func fakeSyncer(syncCh <-chan *syncRequest, done <-chan struct{}) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Returns a processor instance configured for testing purpose. | ||||
| func newProcessorForTest(t *testing.T, r *rdb.RDB, h Handler) *processor { | ||||
| 	starting := make(chan *workerInfo) | ||||
| 	finished := make(chan *base.TaskMessage) | ||||
| 	syncCh := make(chan *syncRequest) | ||||
| 	done := make(chan struct{}) | ||||
| 	t.Cleanup(func() { close(done) }) | ||||
| 	go fakeHeartbeater(starting, finished, done) | ||||
| 	go fakeSyncer(syncCh, done) | ||||
| 	p := newProcessor(processorParams{ | ||||
| 		logger:          testLogger, | ||||
| 		broker:          r, | ||||
| 		retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 		isFailureFunc:   defaultIsFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    base.NewCancelations(), | ||||
| 		concurrency:     10, | ||||
| 		queues:          defaultQueueConfig, | ||||
| 		strictPriority:  false, | ||||
| 		errHandler:      nil, | ||||
| 		shutdownTimeout: defaultShutdownTimeout, | ||||
| 		starting:        starting, | ||||
| 		finished:        finished, | ||||
| 	}) | ||||
| 	p.handler = h | ||||
| 	return p | ||||
| } | ||||
|  | ||||
| func TestProcessorSuccessWithSingleQueue(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
|  | ||||
| 	m1 := h.NewTaskMessage("task1", nil) | ||||
| @@ -86,45 +122,24 @@ func TestProcessorSuccessWithSingleQueue(t *testing.T) { | ||||
| 			processed = append(processed, task) | ||||
| 			return nil | ||||
| 		} | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		syncCh := make(chan *syncRequest) | ||||
| 		done := make(chan struct{}) | ||||
| 		defer func() { close(done) }() | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          defaultQueueConfig, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = HandlerFunc(handler) | ||||
| 		p := newProcessorForTest(t, rdbClient, HandlerFunc(handler)) | ||||
|  | ||||
| 		p.start(&sync.WaitGroup{}) | ||||
| 		for _, msg := range tc.incoming { | ||||
| 			err := rdbClient.Enqueue(msg) | ||||
| 			if err != nil { | ||||
| 				p.terminate() | ||||
| 				p.shutdown() | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 		} | ||||
| 		time.Sleep(2 * time.Second) // wait for two second to allow all pending tasks to be processed. | ||||
| 		if l := r.LLen(base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 		if l := r.LLen(context.Background(), base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 			t.Errorf("%q has %d tasks, want 0", base.ActiveKey(base.DefaultQueueName), l) | ||||
| 		} | ||||
| 		p.terminate() | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		mu.Lock() | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, sortTaskOpt, cmp.AllowUnexported(Payload{})); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, taskCmpOpts...); diff != "" { | ||||
| 			t.Errorf("mismatch found in processed tasks; (-want, +got)\n%s", diff) | ||||
| 		} | ||||
| 		mu.Unlock() | ||||
| @@ -146,6 +161,7 @@ func TestProcessorSuccessWithMultipleQueues(t *testing.T) { | ||||
| 		t3 = NewTask(m3.Type, m3.Payload) | ||||
| 		t4 = NewTask(m4.Type, m4.Payload) | ||||
| 	) | ||||
| 	defer r.Close() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		pending       map[string][]*base.TaskMessage | ||||
| @@ -177,46 +193,26 @@ func TestProcessorSuccessWithMultipleQueues(t *testing.T) { | ||||
| 			processed = append(processed, task) | ||||
| 			return nil | ||||
| 		} | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		syncCh := make(chan *syncRequest) | ||||
| 		done := make(chan struct{}) | ||||
| 		defer func() { close(done) }() | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:         testLogger, | ||||
| 			broker:         rdbClient, | ||||
| 			retryDelayFunc: DefaultRetryDelayFunc, | ||||
| 			syncCh:         syncCh, | ||||
| 			cancelations:   base.NewCancelations(), | ||||
| 			concurrency:    10, | ||||
| 			queues: map[string]int{ | ||||
| 		p := newProcessorForTest(t, rdbClient, HandlerFunc(handler)) | ||||
| 		p.queueConfig = map[string]int{ | ||||
| 			"default": 2, | ||||
| 			"high":    3, | ||||
| 			"low":     1, | ||||
| 			}, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = HandlerFunc(handler) | ||||
| 		} | ||||
|  | ||||
| 		p.start(&sync.WaitGroup{}) | ||||
| 		// Wait for two second to allow all pending tasks to be processed. | ||||
| 		time.Sleep(2 * time.Second) | ||||
| 		// Make sure no messages are stuck in active list. | ||||
| 		for _, qname := range tc.queues { | ||||
| 			if l := r.LLen(base.ActiveKey(qname)).Val(); l != 0 { | ||||
| 			if l := r.LLen(context.Background(), base.ActiveKey(qname)).Val(); l != 0 { | ||||
| 				t.Errorf("%q has %d tasks, want 0", base.ActiveKey(qname), l) | ||||
| 			} | ||||
| 		} | ||||
| 		p.terminate() | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		mu.Lock() | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, sortTaskOpt, cmp.AllowUnexported(Payload{})); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, taskCmpOpts...); diff != "" { | ||||
| 			t.Errorf("mismatch found in processed tasks; (-want, +got)\n%s", diff) | ||||
| 		} | ||||
| 		mu.Unlock() | ||||
| @@ -226,9 +222,10 @@ func TestProcessorSuccessWithMultipleQueues(t *testing.T) { | ||||
| // https://github.com/hibiken/asynq/issues/166 | ||||
| func TestProcessTasksWithLargeNumberInPayload(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
|  | ||||
| 	m1 := h.NewTaskMessage("large_number", map[string]interface{}{"data": 111111111111111111}) | ||||
| 	m1 := h.NewTaskMessage("large_number", h.JSON(map[string]interface{}{"data": 111111111111111111})) | ||||
| 	t1 := NewTask(m1.Type, m1.Payload) | ||||
|  | ||||
| 	tests := []struct { | ||||
| @@ -250,46 +247,29 @@ func TestProcessTasksWithLargeNumberInPayload(t *testing.T) { | ||||
| 		handler := func(ctx context.Context, task *Task) error { | ||||
| 			mu.Lock() | ||||
| 			defer mu.Unlock() | ||||
| 			if data, err := task.Payload.GetInt("data"); err != nil { | ||||
| 				t.Errorf("coult not get data from payload: %v", err) | ||||
| 			} else { | ||||
| 			var payload map[string]int | ||||
| 			if err := json.Unmarshal(task.Payload(), &payload); err != nil { | ||||
| 				t.Errorf("coult not decode payload: %v", err) | ||||
| 			} | ||||
| 			if data, ok := payload["data"]; ok { | ||||
| 				t.Logf("data == %d", data) | ||||
| 			} else { | ||||
| 				t.Errorf("could not get data from payload") | ||||
| 			} | ||||
| 			processed = append(processed, task) | ||||
| 			return nil | ||||
| 		} | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		syncCh := make(chan *syncRequest) | ||||
| 		done := make(chan struct{}) | ||||
| 		defer func() { close(done) }() | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		go fakeSyncer(syncCh, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			syncCh:          syncCh, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          defaultQueueConfig, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = HandlerFunc(handler) | ||||
| 		p := newProcessorForTest(t, rdbClient, HandlerFunc(handler)) | ||||
|  | ||||
| 		p.start(&sync.WaitGroup{}) | ||||
| 		time.Sleep(2 * time.Second) // wait for two second to allow all pending tasks to be processed. | ||||
| 		if l := r.LLen(base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 		if l := r.LLen(context.Background(), base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 			t.Errorf("%q has %d tasks, want 0", base.ActiveKey(base.DefaultQueueName), l) | ||||
| 		} | ||||
| 		p.terminate() | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		mu.Lock() | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, sortTaskOpt, cmpopts.IgnoreUnexported(Payload{})); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, taskCmpOpts...); diff != "" { | ||||
| 			t.Errorf("mismatch found in processed tasks; (-want, +got)\n%s", diff) | ||||
| 		} | ||||
| 		mu.Unlock() | ||||
| @@ -298,6 +278,7 @@ func TestProcessTasksWithLargeNumberInPayload(t *testing.T) { | ||||
|  | ||||
| func TestProcessorRetry(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
|  | ||||
| 	m1 := h.NewTaskMessage("send_email", nil) | ||||
| @@ -308,66 +289,55 @@ func TestProcessorRetry(t *testing.T) { | ||||
|  | ||||
| 	errMsg := "something went wrong" | ||||
| 	wrappedSkipRetry := fmt.Errorf("%s:%w", errMsg, SkipRetry) | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc         string              // test description | ||||
| 		pending      []*base.TaskMessage // initial default queue state | ||||
| 		incoming     []*base.TaskMessage // tasks to be enqueued during run | ||||
| 		delay        time.Duration       // retry delay duration | ||||
| 		handler      Handler             // task handler | ||||
| 		wait         time.Duration       // wait duration between starting and stopping processor for this test case | ||||
| 		wantRetry    []base.Z            // tasks in retry queue at the end | ||||
| 		wantErrMsg   string              // error message the task should record | ||||
| 		wantRetry    []*base.TaskMessage // tasks in retry queue at the end | ||||
| 		wantArchived []*base.TaskMessage // tasks in archived queue at the end | ||||
| 		wantErrCount int                 // number of times error handler should be called | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:    "Should automatically retry errored tasks", | ||||
| 			pending:  []*base.TaskMessage{m1, m2}, | ||||
| 			incoming: []*base.TaskMessage{m3, m4}, | ||||
| 			pending: []*base.TaskMessage{m1, m2, m3, m4}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return fmt.Errorf(errMsg) | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantRetry: []base.Z{ | ||||
| 				{Message: h.TaskMessageAfterRetry(*m2, errMsg), Score: now.Add(time.Minute).Unix()}, | ||||
| 				{Message: h.TaskMessageAfterRetry(*m3, errMsg), Score: now.Add(time.Minute).Unix()}, | ||||
| 				{Message: h.TaskMessageAfterRetry(*m4, errMsg), Score: now.Add(time.Minute).Unix()}, | ||||
| 			}, | ||||
| 			wantArchived: []*base.TaskMessage{h.TaskMessageWithError(*m1, errMsg)}, | ||||
| 			wantErrMsg:   errMsg, | ||||
| 			wantRetry:    []*base.TaskMessage{m2, m3, m4}, | ||||
| 			wantArchived: []*base.TaskMessage{m1}, | ||||
| 			wantErrCount: 4, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should skip retry errored tasks", | ||||
| 			pending: []*base.TaskMessage{m1, m2}, | ||||
| 			incoming: []*base.TaskMessage{}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return SkipRetry // return SkipRetry without wrapping | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantRetry: []base.Z{}, | ||||
| 			wantArchived: []*base.TaskMessage{ | ||||
| 				h.TaskMessageWithError(*m1, SkipRetry.Error()), | ||||
| 				h.TaskMessageWithError(*m2, SkipRetry.Error()), | ||||
| 			}, | ||||
| 			wantErrMsg:   SkipRetry.Error(), | ||||
| 			wantRetry:    []*base.TaskMessage{}, | ||||
| 			wantArchived: []*base.TaskMessage{m1, m2}, | ||||
| 			wantErrCount: 2, // ErrorHandler should still be called with SkipRetry error | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should skip retry errored tasks (with error wrapping)", | ||||
| 			pending: []*base.TaskMessage{m1, m2}, | ||||
| 			incoming: []*base.TaskMessage{}, | ||||
| 			delay:   time.Minute, | ||||
| 			handler: HandlerFunc(func(ctx context.Context, task *Task) error { | ||||
| 				return wrappedSkipRetry | ||||
| 			}), | ||||
| 			wait:         2 * time.Second, | ||||
| 			wantRetry: []base.Z{}, | ||||
| 			wantArchived: []*base.TaskMessage{ | ||||
| 				h.TaskMessageWithError(*m1, wrappedSkipRetry.Error()), | ||||
| 				h.TaskMessageWithError(*m2, wrappedSkipRetry.Error()), | ||||
| 			}, | ||||
| 			wantErrMsg:   wrappedSkipRetry.Error(), | ||||
| 			wantRetry:    []*base.TaskMessage{}, | ||||
| 			wantArchived: []*base.TaskMessage{m1, m2}, | ||||
| 			wantErrCount: 2, // ErrorHandler should still be called with SkipRetry error | ||||
| 		}, | ||||
| 	} | ||||
| @@ -389,50 +359,43 @@ func TestProcessorRetry(t *testing.T) { | ||||
| 			defer mu.Unlock() | ||||
| 			n++ | ||||
| 		} | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		done := make(chan struct{}) | ||||
| 		defer func() { close(done) }() | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			retryDelayFunc:  delayFunc, | ||||
| 			syncCh:          nil, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          defaultQueueConfig, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      ErrorHandlerFunc(errHandler), | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		p.handler = tc.handler | ||||
| 		p := newProcessorForTest(t, rdbClient, tc.handler) | ||||
| 		p.errHandler = ErrorHandlerFunc(errHandler) | ||||
| 		p.retryDelayFunc = delayFunc | ||||
|  | ||||
| 		p.start(&sync.WaitGroup{}) | ||||
| 		for _, msg := range tc.incoming { | ||||
| 			err := rdbClient.Enqueue(msg) | ||||
| 			if err != nil { | ||||
| 				p.terminate() | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 		} | ||||
| 		runTime := time.Now() // time when processor is running | ||||
| 		time.Sleep(tc.wait)   // FIXME: This makes test flaky. | ||||
| 		p.terminate() | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		cmpOpt := h.EquateInt64Approx(1) // allow up to a second difference in zset score | ||||
| 		cmpOpt := h.EquateInt64Approx(int64(tc.wait.Seconds())) // allow up to a wait-second difference in zset score | ||||
| 		gotRetry := h.GetRetryEntries(t, r, base.DefaultQueueName) | ||||
| 		if diff := cmp.Diff(tc.wantRetry, gotRetry, h.SortZSetEntryOpt, cmpOpt); diff != "" { | ||||
| 		var wantRetry []base.Z // Note: construct wantRetry here since `LastFailedAt` and ZSCORE is relative to each test run. | ||||
| 		for _, msg := range tc.wantRetry { | ||||
| 			wantRetry = append(wantRetry, | ||||
| 				base.Z{ | ||||
| 					Message: h.TaskMessageAfterRetry(*msg, tc.wantErrMsg, runTime), | ||||
| 					Score:   runTime.Add(tc.delay).Unix(), | ||||
| 				}) | ||||
| 		} | ||||
| 		if diff := cmp.Diff(wantRetry, gotRetry, h.SortZSetEntryOpt, cmpOpt); diff != "" { | ||||
| 			t.Errorf("%s: mismatch found in %q after running processor; (-want, +got)\n%s", tc.desc, base.RetryKey(base.DefaultQueueName), diff) | ||||
| 		} | ||||
|  | ||||
| 		gotDead := h.GetArchivedMessages(t, r, base.DefaultQueueName) | ||||
| 		if diff := cmp.Diff(tc.wantArchived, gotDead, h.SortMsgOpt); diff != "" { | ||||
| 		gotArchived := h.GetArchivedEntries(t, r, base.DefaultQueueName) | ||||
| 		var wantArchived []base.Z // Note: construct wantArchived here since `LastFailedAt` and ZSCORE is relative to each test run. | ||||
| 		for _, msg := range tc.wantArchived { | ||||
| 			wantArchived = append(wantArchived, | ||||
| 				base.Z{ | ||||
| 					Message: h.TaskMessageWithError(*msg, tc.wantErrMsg, runTime), | ||||
| 					Score:   runTime.Unix(), | ||||
| 				}) | ||||
| 		} | ||||
| 		if diff := cmp.Diff(wantArchived, gotArchived, h.SortZSetEntryOpt, cmpOpt); diff != "" { | ||||
| 			t.Errorf("%s: mismatch found in %q after running processor; (-want, +got)\n%s", tc.desc, base.ArchivedKey(base.DefaultQueueName), diff) | ||||
| 		} | ||||
|  | ||||
| 		if l := r.LLen(base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 		if l := r.LLen(context.Background(), base.ActiveKey(base.DefaultQueueName)).Val(); l != 0 { | ||||
| 			t.Errorf("%s: %q has %d tasks, want 0", base.ActiveKey(base.DefaultQueueName), tc.desc, l) | ||||
| 		} | ||||
|  | ||||
| @@ -442,6 +405,81 @@ func TestProcessorRetry(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestProcessorMarkAsComplete(t *testing.T) { | ||||
| 	r := setup(t) | ||||
| 	defer r.Close() | ||||
| 	rdbClient := rdb.NewRDB(r) | ||||
|  | ||||
| 	msg1 := h.NewTaskMessage("one", nil) | ||||
| 	msg2 := h.NewTaskMessage("two", nil) | ||||
| 	msg3 := h.NewTaskMessageWithQueue("three", nil, "custom") | ||||
| 	msg1.Retention = 3600 | ||||
| 	msg3.Retention = 7200 | ||||
|  | ||||
| 	handler := func(ctx context.Context, task *Task) error { return nil } | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		pending       map[string][]*base.TaskMessage | ||||
| 		completed     map[string][]base.Z | ||||
| 		queueCfg      map[string]int | ||||
| 		wantPending   map[string][]*base.TaskMessage | ||||
| 		wantCompleted func(completedAt time.Time) map[string][]base.Z | ||||
| 	}{ | ||||
| 		{ | ||||
| 			pending: map[string][]*base.TaskMessage{ | ||||
| 				"default": {msg1, msg2}, | ||||
| 				"custom":  {msg3}, | ||||
| 			}, | ||||
| 			completed: map[string][]base.Z{ | ||||
| 				"default": {}, | ||||
| 				"custom":  {}, | ||||
| 			}, | ||||
| 			queueCfg: map[string]int{ | ||||
| 				"default": 1, | ||||
| 				"custom":  1, | ||||
| 			}, | ||||
| 			wantPending: map[string][]*base.TaskMessage{ | ||||
| 				"default": {}, | ||||
| 				"custom":  {}, | ||||
| 			}, | ||||
| 			wantCompleted: func(completedAt time.Time) map[string][]base.Z { | ||||
| 				return map[string][]base.Z{ | ||||
| 					"default": {{Message: h.TaskMessageWithCompletedAt(*msg1, completedAt), Score: completedAt.Unix() + msg1.Retention}}, | ||||
| 					"custom":  {{Message: h.TaskMessageWithCompletedAt(*msg3, completedAt), Score: completedAt.Unix() + msg3.Retention}}, | ||||
| 				} | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		h.FlushDB(t, r) | ||||
| 		h.SeedAllPendingQueues(t, r, tc.pending) | ||||
| 		h.SeedAllCompletedQueues(t, r, tc.completed) | ||||
|  | ||||
| 		p := newProcessorForTest(t, rdbClient, HandlerFunc(handler)) | ||||
| 		p.queueConfig = tc.queueCfg | ||||
|  | ||||
| 		p.start(&sync.WaitGroup{}) | ||||
| 		runTime := time.Now() // time when processor is running | ||||
| 		time.Sleep(2 * time.Second) | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		for qname, want := range tc.wantPending { | ||||
| 			gotPending := h.GetPendingMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotPending, cmpopts.EquateEmpty()); diff != "" { | ||||
| 				t.Errorf("diff found in %q pending set; want=%v, got=%v\n%s", qname, want, gotPending, diff) | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		for qname, want := range tc.wantCompleted(runTime) { | ||||
| 			gotCompleted := h.GetCompletedEntries(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotCompleted, cmpopts.EquateEmpty()); diff != "" { | ||||
| 				t.Errorf("diff found in %q completed set; want=%v, got=%v\n%s", qname, want, gotCompleted, diff) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestProcessorQueues(t *testing.T) { | ||||
| 	sortOpt := cmp.Transformer("SortStrings", func(in []string) []string { | ||||
| 		out := append([]string(nil), in...) // Copy input to avoid mutating it | ||||
| @@ -470,25 +508,10 @@ func TestProcessorQueues(t *testing.T) { | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		done := make(chan struct{}) | ||||
| 		defer func() { close(done) }() | ||||
| 		go fakeHeartbeater(starting, finished, done) | ||||
| 		p := newProcessor(processorParams{ | ||||
| 			logger:          testLogger, | ||||
| 			broker:          nil, | ||||
| 			retryDelayFunc:  DefaultRetryDelayFunc, | ||||
| 			syncCh:          nil, | ||||
| 			cancelations:    base.NewCancelations(), | ||||
| 			concurrency:     10, | ||||
| 			queues:          tc.queueCfg, | ||||
| 			strictPriority:  false, | ||||
| 			errHandler:      nil, | ||||
| 			shutdownTimeout: defaultShutdownTimeout, | ||||
| 			starting:        starting, | ||||
| 			finished:        finished, | ||||
| 		}) | ||||
| 		// Note: rdb and handler not needed for this test. | ||||
| 		p := newProcessorForTest(t, nil, nil) | ||||
| 		p.queueConfig = tc.queueCfg | ||||
|  | ||||
| 		got := p.queues() | ||||
| 		if diff := cmp.Diff(tc.want, got, sortOpt); diff != "" { | ||||
| 			t.Errorf("with queue config: %v\n(*processor).queues() = %v, want %v\n(-want,+got):\n%s", | ||||
| @@ -559,7 +582,7 @@ func TestProcessorWithStrictPriority(t *testing.T) { | ||||
| 			"critical":            3, | ||||
| 			"low":                 1, | ||||
| 		} | ||||
| 		starting := make(chan *base.TaskMessage) | ||||
| 		starting := make(chan *workerInfo) | ||||
| 		finished := make(chan *base.TaskMessage) | ||||
| 		syncCh := make(chan *syncRequest) | ||||
| 		done := make(chan struct{}) | ||||
| @@ -570,6 +593,7 @@ func TestProcessorWithStrictPriority(t *testing.T) { | ||||
| 			logger:          testLogger, | ||||
| 			broker:          rdbClient, | ||||
| 			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. | ||||
| @@ -586,13 +610,13 @@ func TestProcessorWithStrictPriority(t *testing.T) { | ||||
| 		time.Sleep(tc.wait) | ||||
| 		// Make sure no tasks are stuck in active list. | ||||
| 		for _, qname := range tc.queues { | ||||
| 			if l := r.LLen(base.ActiveKey(qname)).Val(); l != 0 { | ||||
| 			if l := r.LLen(context.Background(), base.ActiveKey(qname)).Val(); l != 0 { | ||||
| 				t.Errorf("%q has %d tasks, want 0", base.ActiveKey(qname), l) | ||||
| 			} | ||||
| 		} | ||||
| 		p.terminate() | ||||
| 		p.shutdown() | ||||
|  | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, cmp.AllowUnexported(Payload{})); diff != "" { | ||||
| 		if diff := cmp.Diff(tc.wantProcessed, processed, taskCmpOpts...); diff != "" { | ||||
| 			t.Errorf("mismatch found in processed tasks; (-want, +got)\n%s", diff) | ||||
| 		} | ||||
|  | ||||
| @@ -611,7 +635,7 @@ func TestProcessorPerform(t *testing.T) { | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				return nil | ||||
| 			}, | ||||
| 			task:    NewTask("gen_thumbnail", map[string]interface{}{"src": "some/img/path"}), | ||||
| 			task:    NewTask("gen_thumbnail", h.JSON(map[string]interface{}{"src": "some/img/path"})), | ||||
| 			wantErr: false, | ||||
| 		}, | ||||
| 		{ | ||||
| @@ -619,7 +643,7 @@ func TestProcessorPerform(t *testing.T) { | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				return fmt.Errorf("something went wrong") | ||||
| 			}, | ||||
| 			task:    NewTask("gen_thumbnail", map[string]interface{}{"src": "some/img/path"}), | ||||
| 			task:    NewTask("gen_thumbnail", h.JSON(map[string]interface{}{"src": "some/img/path"})), | ||||
| 			wantErr: true, | ||||
| 		}, | ||||
| 		{ | ||||
| @@ -627,16 +651,13 @@ func TestProcessorPerform(t *testing.T) { | ||||
| 			handler: func(ctx context.Context, t *Task) error { | ||||
| 				panic("something went terribly wrong") | ||||
| 			}, | ||||
| 			task:    NewTask("gen_thumbnail", map[string]interface{}{"src": "some/img/path"}), | ||||
| 			task:    NewTask("gen_thumbnail", h.JSON(map[string]interface{}{"src": "some/img/path"})), | ||||
| 			wantErr: true, | ||||
| 		}, | ||||
| 	} | ||||
| 	// Note: We don't need to fully initialize the processor since we are only testing | ||||
| 	// Note: We don't need to fully initialized the processor since we are only testing | ||||
| 	// perform method. | ||||
| 	p := newProcessor(processorParams{ | ||||
| 		logger: testLogger, | ||||
| 		queues: defaultQueueConfig, | ||||
| 	}) | ||||
| 	p := newProcessorForTest(t, nil, nil) | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		p.handler = tc.handler | ||||
|   | ||||
							
								
								
									
										53
									
								
								recoverer.go
									
									
									
									
									
								
							
							
						
						
									
										53
									
								
								recoverer.go
									
									
									
									
									
								
							| @@ -5,7 +5,7 @@ | ||||
| package asynq | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"context" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| @@ -17,6 +17,7 @@ type recoverer struct { | ||||
| 	logger         *log.Logger | ||||
| 	broker         base.Broker | ||||
| 	retryDelayFunc RetryDelayFunc | ||||
| 	isFailureFunc  func(error) bool | ||||
|  | ||||
| 	// channel to communicate back to the long running "recoverer" goroutine. | ||||
| 	done chan struct{} | ||||
| @@ -34,6 +35,7 @@ type recovererParams struct { | ||||
| 	queues         []string | ||||
| 	interval       time.Duration | ||||
| 	retryDelayFunc RetryDelayFunc | ||||
| 	isFailureFunc  func(error) bool | ||||
| } | ||||
|  | ||||
| func newRecoverer(params recovererParams) *recoverer { | ||||
| @@ -44,10 +46,11 @@ func newRecoverer(params recovererParams) *recoverer { | ||||
| 		queues:         params.queues, | ||||
| 		interval:       params.interval, | ||||
| 		retryDelayFunc: params.retryDelayFunc, | ||||
| 		isFailureFunc:  params.isFailureFunc, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (r *recoverer) terminate() { | ||||
| func (r *recoverer) shutdown() { | ||||
| 	r.logger.Debug("Recoverer shutting down...") | ||||
| 	// Signal the recoverer goroutine to stop polling. | ||||
| 	r.done <- struct{}{} | ||||
| @@ -57,6 +60,7 @@ func (r *recoverer) start(wg *sync.WaitGroup) { | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		r.recover() | ||||
| 		timer := time.NewTimer(r.interval) | ||||
| 		for { | ||||
| 			select { | ||||
| @@ -65,37 +69,40 @@ func (r *recoverer) start(wg *sync.WaitGroup) { | ||||
| 				timer.Stop() | ||||
| 				return | ||||
| 			case <-timer.C: | ||||
| 				// Get all tasks which have expired 30 seconds ago or earlier. | ||||
| 				deadline := time.Now().Add(-30 * time.Second) | ||||
| 				msgs, err := r.broker.ListDeadlineExceeded(deadline, r.queues...) | ||||
| 				if err != nil { | ||||
| 					r.logger.Warn("recoverer: could not list deadline exceeded tasks") | ||||
| 					continue | ||||
| 				} | ||||
| 				const errMsg = "deadline exceeded" // TODO: better error message | ||||
| 				for _, msg := range msgs { | ||||
| 					if msg.Retried >= msg.Retry { | ||||
| 						r.archive(msg, errMsg) | ||||
| 					} else { | ||||
| 						r.retry(msg, errMsg) | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				r.recover() | ||||
| 				timer.Reset(r.interval) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| } | ||||
|  | ||||
| func (r *recoverer) retry(msg *base.TaskMessage, errMsg string) { | ||||
| 	delay := r.retryDelayFunc(msg.Retried, fmt.Errorf(errMsg), NewTask(msg.Type, msg.Payload)) | ||||
| func (r *recoverer) recover() { | ||||
| 	// Get all tasks which have expired 30 seconds ago or earlier. | ||||
| 	deadline := time.Now().Add(-30 * time.Second) | ||||
| 	msgs, err := r.broker.ListDeadlineExceeded(deadline, r.queues...) | ||||
| 	if err != nil { | ||||
| 		r.logger.Warn("recoverer: could not list deadline exceeded tasks") | ||||
| 		return | ||||
| 	} | ||||
| 	for _, msg := range msgs { | ||||
| 		if msg.Retried >= msg.Retry { | ||||
| 			r.archive(msg, context.DeadlineExceeded) | ||||
| 		} else { | ||||
| 			r.retry(msg, context.DeadlineExceeded) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (r *recoverer) retry(msg *base.TaskMessage, err error) { | ||||
| 	delay := r.retryDelayFunc(msg.Retried, err, NewTask(msg.Type, msg.Payload)) | ||||
| 	retryAt := time.Now().Add(delay) | ||||
| 	if err := r.broker.Retry(msg, retryAt, errMsg); err != nil { | ||||
| 	if err := r.broker.Retry(msg, retryAt, err.Error(), r.isFailureFunc(err)); err != nil { | ||||
| 		r.logger.Warnf("recoverer: could not retry deadline exceeded task: %v", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (r *recoverer) archive(msg *base.TaskMessage, errMsg string) { | ||||
| 	if err := r.broker.Archive(msg, errMsg); err != nil { | ||||
| func (r *recoverer) archive(msg *base.TaskMessage, err error) { | ||||
| 	if err := r.broker.Archive(msg, err.Error()); err != nil { | ||||
| 		r.logger.Warnf("recoverer: could not move task to archive: %v", err) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -64,7 +64,7 @@ func TestRecoverer(t *testing.T) { | ||||
| 				"default": {}, | ||||
| 			}, | ||||
| 			wantRetry: map[string][]*base.TaskMessage{ | ||||
| 				"default": {h.TaskMessageAfterRetry(*t1, "deadline exceeded")}, | ||||
| 				"default": {t1}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]*base.TaskMessage{ | ||||
| 				"default": {}, | ||||
| @@ -101,7 +101,7 @@ func TestRecoverer(t *testing.T) { | ||||
| 				"critical": {}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]*base.TaskMessage{ | ||||
| 				"default":  {h.TaskMessageWithError(*t4, "deadline exceeded")}, | ||||
| 				"default":  {t4}, | ||||
| 				"critical": {}, | ||||
| 			}, | ||||
| 		}, | ||||
| @@ -137,7 +137,7 @@ func TestRecoverer(t *testing.T) { | ||||
| 				"critical": {{Message: t3, Score: oneHourFromNow.Unix()}}, | ||||
| 			}, | ||||
| 			wantRetry: map[string][]*base.TaskMessage{ | ||||
| 				"default":  {h.TaskMessageAfterRetry(*t1, "deadline exceeded")}, | ||||
| 				"default":  {t1}, | ||||
| 				"critical": {}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]*base.TaskMessage{ | ||||
| @@ -176,8 +176,8 @@ func TestRecoverer(t *testing.T) { | ||||
| 				"default": {{Message: t2, Score: oneHourFromNow.Unix()}}, | ||||
| 			}, | ||||
| 			wantRetry: map[string][]*base.TaskMessage{ | ||||
| 				"default":  {h.TaskMessageAfterRetry(*t1, "deadline exceeded")}, | ||||
| 				"critical": {h.TaskMessageAfterRetry(*t3, "deadline exceeded")}, | ||||
| 				"default":  {t1}, | ||||
| 				"critical": {t3}, | ||||
| 			}, | ||||
| 			wantArchived: map[string][]*base.TaskMessage{ | ||||
| 				"default":  {}, | ||||
| @@ -234,12 +234,14 @@ func TestRecoverer(t *testing.T) { | ||||
| 			queues:         []string{"default", "critical"}, | ||||
| 			interval:       1 * time.Second, | ||||
| 			retryDelayFunc: func(n int, err error, task *Task) time.Duration { return 30 * time.Second }, | ||||
| 			isFailureFunc:  defaultIsFailureFunc, | ||||
| 		}) | ||||
|  | ||||
| 		var wg sync.WaitGroup | ||||
| 		recoverer.start(&wg) | ||||
| 		runTime := time.Now() // time when recoverer is running | ||||
| 		time.Sleep(2 * time.Second) | ||||
| 		recoverer.terminate() | ||||
| 		recoverer.shutdown() | ||||
|  | ||||
| 		for qname, want := range tc.wantActive { | ||||
| 			gotActive := h.GetActiveMessages(t, r, qname) | ||||
| @@ -253,15 +255,24 @@ func TestRecoverer(t *testing.T) { | ||||
| 				t.Errorf("%s; mismatch found in %q; (-want,+got)\n%s", tc.desc, base.DeadlinesKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 		for qname, want := range tc.wantRetry { | ||||
| 		cmpOpt := h.EquateInt64Approx(2) // allow up to two-second difference in `LastFailedAt` | ||||
| 		for qname, msgs := range tc.wantRetry { | ||||
| 			gotRetry := h.GetRetryMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotRetry, h.SortMsgOpt); diff != "" { | ||||
| 			var wantRetry []*base.TaskMessage // Note: construct message here since `LastFailedAt` is relative to each test run | ||||
| 			for _, msg := range msgs { | ||||
| 				wantRetry = append(wantRetry, h.TaskMessageAfterRetry(*msg, "context deadline exceeded", runTime)) | ||||
| 			} | ||||
| 			if diff := cmp.Diff(wantRetry, gotRetry, h.SortMsgOpt, cmpOpt); diff != "" { | ||||
| 				t.Errorf("%s; mismatch found in %q: (-want, +got)\n%s", tc.desc, base.RetryKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
| 		for qname, want := range tc.wantArchived { | ||||
| 			gotDead := h.GetArchivedMessages(t, r, qname) | ||||
| 			if diff := cmp.Diff(want, gotDead, h.SortMsgOpt); diff != "" { | ||||
| 		for qname, msgs := range tc.wantArchived { | ||||
| 			gotArchived := h.GetArchivedMessages(t, r, qname) | ||||
| 			var wantArchived []*base.TaskMessage | ||||
| 			for _, msg := range msgs { | ||||
| 				wantArchived = append(wantArchived, h.TaskMessageWithError(*msg, "context deadline exceeded", runTime)) | ||||
| 			} | ||||
| 			if diff := cmp.Diff(wantArchived, gotArchived, h.SortMsgOpt, cmpOpt); diff != "" { | ||||
| 				t.Errorf("%s; mismatch found in %q: (-want, +got)\n%s", tc.desc, base.ArchivedKey(qname), diff) | ||||
| 			} | ||||
| 		} | ||||
|   | ||||
							
								
								
									
										89
									
								
								scheduler.go
									
									
									
									
									
								
							
							
						
						
									
										89
									
								
								scheduler.go
									
									
									
									
									
								
							| @@ -10,6 +10,7 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| @@ -18,9 +19,11 @@ import ( | ||||
| ) | ||||
|  | ||||
| // A Scheduler kicks off tasks at regular intervals based on the user defined schedule. | ||||
| // | ||||
| // Schedulers are safe for concurrent use by multiple goroutines. | ||||
| type Scheduler struct { | ||||
| 	id         string | ||||
| 	status     *base.ServerStatus | ||||
| 	state      *base.ServerState | ||||
| 	logger     *log.Logger | ||||
| 	client     *Client | ||||
| 	rdb        *rdb.RDB | ||||
| @@ -29,11 +32,22 @@ type Scheduler struct { | ||||
| 	done       chan struct{} | ||||
| 	wg         sync.WaitGroup | ||||
| 	errHandler func(task *Task, opts []Option, err error) | ||||
|  | ||||
| 	// guards idmap | ||||
| 	mu sync.Mutex | ||||
| 	// idmap maps Scheduler's entry ID to cron.EntryID | ||||
| 	// to avoid using cron.EntryID as the public API of | ||||
| 	// the Scheduler. | ||||
| 	idmap map[string]cron.EntryID | ||||
| } | ||||
|  | ||||
| // 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) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	if opts == nil { | ||||
| 		opts = &SchedulerOpts{} | ||||
| 	} | ||||
| @@ -52,14 +66,15 @@ func NewScheduler(r RedisConnOpt, opts *SchedulerOpts) *Scheduler { | ||||
|  | ||||
| 	return &Scheduler{ | ||||
| 		id:         generateSchedulerID(), | ||||
| 		status:     base.NewServerStatus(base.StatusIdle), | ||||
| 		state:      base.NewServerState(), | ||||
| 		logger:     logger, | ||||
| 		client:     NewClient(r), | ||||
| 		rdb:        rdb.NewRDB(createRedisClient(r)), | ||||
| 		rdb:        rdb.NewRDB(c), | ||||
| 		cron:       cron.New(cron.WithLocation(loc)), | ||||
| 		location:   loc, | ||||
| 		done:       make(chan struct{}), | ||||
| 		errHandler: opts.EnqueueErrorHandler, | ||||
| 		idmap:      make(map[string]cron.EntryID), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -107,7 +122,7 @@ type enqueueJob struct { | ||||
| } | ||||
|  | ||||
| func (j *enqueueJob) Run() { | ||||
| 	res, err := j.client.Enqueue(j.task, j.opts...) | ||||
| 	info, err := j.client.Enqueue(j.task, j.opts...) | ||||
| 	if err != nil { | ||||
| 		j.logger.Errorf("scheduler could not enqueue a task %+v: %v", j.task, err) | ||||
| 		if j.errHandler != nil { | ||||
| @@ -115,10 +130,10 @@ func (j *enqueueJob) Run() { | ||||
| 		} | ||||
| 		return | ||||
| 	} | ||||
| 	j.logger.Debugf("scheduler enqueued a task: %+v", res) | ||||
| 	j.logger.Debugf("scheduler enqueued a task: %+v", info) | ||||
| 	event := &base.SchedulerEnqueueEvent{ | ||||
| 		TaskID:     res.ID, | ||||
| 		EnqueuedAt: res.EnqueuedAt.In(j.location), | ||||
| 		TaskID:     info.ID, | ||||
| 		EnqueuedAt: time.Now().In(j.location), | ||||
| 	} | ||||
| 	err = j.rdb.RecordSchedulerEnqueueEvent(j.id.String(), event) | ||||
| 	if err != nil { | ||||
| @@ -140,29 +155,48 @@ func (s *Scheduler) Register(cronspec string, task *Task, opts ...Option) (entry | ||||
| 		logger:     s.logger, | ||||
| 		errHandler: s.errHandler, | ||||
| 	} | ||||
| 	if _, err = s.cron.AddJob(cronspec, job); err != nil { | ||||
| 	cronID, err := s.cron.AddJob(cronspec, job) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	s.mu.Lock() | ||||
| 	s.idmap[job.id.String()] = cronID | ||||
| 	s.mu.Unlock() | ||||
| 	return job.id.String(), nil | ||||
| } | ||||
|  | ||||
| // Unregister removes a registered entry by entry ID. | ||||
| // Unregister returns a non-nil error if no entries were found for the given entryID. | ||||
| func (s *Scheduler) Unregister(entryID string) error { | ||||
| 	s.mu.Lock() | ||||
| 	defer s.mu.Unlock() | ||||
| 	cronID, ok := s.idmap[entryID] | ||||
| 	if !ok { | ||||
| 		return fmt.Errorf("asynq: no scheduler entry found") | ||||
| 	} | ||||
| 	delete(s.idmap, entryID) | ||||
| 	s.cron.Remove(cronID) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Run starts the scheduler until an os signal to exit the program is received. | ||||
| // It returns an error if scheduler is already running or has been stopped. | ||||
| // It returns an error if scheduler is already running or has been shutdown. | ||||
| func (s *Scheduler) Run() error { | ||||
| 	if err := s.Start(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	s.waitForSignals() | ||||
| 	return s.Stop() | ||||
| 	s.Shutdown() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Start starts the scheduler. | ||||
| // It returns an error if the scheduler is already running or has been stopped. | ||||
| // It returns an error if the scheduler is already running or has been shutdown. | ||||
| func (s *Scheduler) Start() error { | ||||
| 	switch s.status.Get() { | ||||
| 	case base.StatusRunning: | ||||
| 	switch s.state.Get() { | ||||
| 	case base.StateActive: | ||||
| 		return fmt.Errorf("asynq: the scheduler is already running") | ||||
| 	case base.StatusStopped: | ||||
| 	case base.StateClosed: | ||||
| 		return fmt.Errorf("asynq: the scheduler has already been stopped") | ||||
| 	} | ||||
| 	s.logger.Info("Scheduler starting") | ||||
| @@ -170,27 +204,23 @@ func (s *Scheduler) Start() error { | ||||
| 	s.cron.Start() | ||||
| 	s.wg.Add(1) | ||||
| 	go s.runHeartbeater() | ||||
| 	s.status.Set(base.StatusRunning) | ||||
| 	s.state.Set(base.StateActive) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Stop stops the scheduler. | ||||
| // It returns an error if the scheduler is not currently running. | ||||
| func (s *Scheduler) Stop() error { | ||||
| 	if s.status.Get() != base.StatusRunning { | ||||
| 		return fmt.Errorf("asynq: the scheduler is not running") | ||||
| 	} | ||||
| // Shutdown stops and shuts down the scheduler. | ||||
| func (s *Scheduler) Shutdown() { | ||||
| 	s.logger.Info("Scheduler shutting down") | ||||
| 	close(s.done) // signal heartbeater to stop | ||||
| 	ctx := s.cron.Stop() | ||||
| 	<-ctx.Done() | ||||
| 	s.wg.Wait() | ||||
|  | ||||
| 	s.clearHistory() | ||||
| 	s.client.Close() | ||||
| 	s.rdb.Close() | ||||
| 	s.status.Set(base.StatusStopped) | ||||
| 	s.state.Set(base.StateClosed) | ||||
| 	s.logger.Info("Scheduler stopped") | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *Scheduler) runHeartbeater() { | ||||
| @@ -216,8 +246,8 @@ func (s *Scheduler) beat() { | ||||
| 		e := &base.SchedulerEntry{ | ||||
| 			ID:      job.id.String(), | ||||
| 			Spec:    job.cronspec, | ||||
| 			Type:    job.task.Type, | ||||
| 			Payload: job.task.Payload.data, | ||||
| 			Type:    job.task.Type(), | ||||
| 			Payload: job.task.Payload(), | ||||
| 			Opts:    stringifyOptions(job.opts), | ||||
| 			Next:    entry.Next, | ||||
| 			Prev:    entry.Prev, | ||||
| @@ -237,3 +267,12 @@ func stringifyOptions(opts []Option) []string { | ||||
| 	} | ||||
| 	return res | ||||
| } | ||||
|  | ||||
| func (s *Scheduler) clearHistory() { | ||||
| 	for _, entry := range s.cron.Entries() { | ||||
| 		job := entry.Job.(*enqueueJob) | ||||
| 		if err := s.rdb.ClearSchedulerHistory(job.id.String()); err != nil { | ||||
| 			s.logger.Warnf("Could not clear scheduler history for entry %q: %v", job.id.String(), err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -14,7 +14,7 @@ import ( | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| ) | ||||
|  | ||||
| func TestScheduler(t *testing.T) { | ||||
| func TestSchedulerRegister(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		cronspec string | ||||
| 		task     *Task | ||||
| @@ -67,9 +67,7 @@ func TestScheduler(t *testing.T) { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		time.Sleep(tc.wait) | ||||
| 		if err := scheduler.Stop(); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		scheduler.Shutdown() | ||||
|  | ||||
| 		got := asynqtest.GetPendingMessages(t, r, tc.queue) | ||||
| 		if diff := cmp.Diff(tc.want, got, asynqtest.IgnoreIDOpt); diff != "" { | ||||
| @@ -106,9 +104,7 @@ func TestSchedulerWhenRedisDown(t *testing.T) { | ||||
| 	} | ||||
| 	// Scheduler should attempt to enqueue the task three times (every 3s). | ||||
| 	time.Sleep(10 * time.Second) | ||||
| 	if err := scheduler.Stop(); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	scheduler.Shutdown() | ||||
|  | ||||
| 	mu.Lock() | ||||
| 	if counter != 3 { | ||||
| @@ -116,3 +112,45 @@ func TestSchedulerWhenRedisDown(t *testing.T) { | ||||
| 	} | ||||
| 	mu.Unlock() | ||||
| } | ||||
|  | ||||
| func TestSchedulerUnregister(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		cronspec string | ||||
| 		task     *Task | ||||
| 		opts     []Option | ||||
| 		wait     time.Duration | ||||
| 		queue    string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			cronspec: "@every 3s", | ||||
| 			task:     NewTask("task1", nil), | ||||
| 			opts:     []Option{MaxRetry(10)}, | ||||
| 			wait:     10 * time.Second, | ||||
| 			queue:    "default", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	r := setup(t) | ||||
|  | ||||
| 	for _, tc := range tests { | ||||
| 		scheduler := NewScheduler(getRedisConnOpt(t), nil) | ||||
| 		entryID, err := scheduler.Register(tc.cronspec, tc.task, tc.opts...) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		if err := scheduler.Unregister(entryID); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		if err := scheduler.Start(); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		time.Sleep(tc.wait) | ||||
| 		scheduler.Shutdown() | ||||
|  | ||||
| 		got := asynqtest.GetPendingMessages(t, r, tc.queue) | ||||
| 		if len(got) != 0 { | ||||
| 			t.Errorf("%d tasks were enqueued, want zero", len(got)) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -62,7 +62,7 @@ func (mux *ServeMux) Handler(t *Task) (h Handler, pattern string) { | ||||
| 	mux.mu.RLock() | ||||
| 	defer mux.mu.RUnlock() | ||||
|  | ||||
| 	h, pattern = mux.match(t.Type) | ||||
| 	h, pattern = mux.match(t.Type()) | ||||
| 	if h == nil { | ||||
| 		h, pattern = NotFoundHandler(), "" | ||||
| 	} | ||||
| @@ -98,7 +98,7 @@ func (mux *ServeMux) Handle(pattern string, handler Handler) { | ||||
| 	mux.mu.Lock() | ||||
| 	defer mux.mu.Unlock() | ||||
|  | ||||
| 	if pattern == "" { | ||||
| 	if strings.TrimSpace(pattern) == "" { | ||||
| 		panic("asynq: invalid pattern") | ||||
| 	} | ||||
| 	if handler == nil { | ||||
| @@ -151,7 +151,7 @@ func (mux *ServeMux) Use(mws ...MiddlewareFunc) { | ||||
|  | ||||
| // NotFound returns an error indicating that the handler was not found for the given task. | ||||
| func NotFound(ctx context.Context, task *Task) error { | ||||
| 	return fmt.Errorf("handler not found for task %q", task.Type) | ||||
| 	return fmt.Errorf("handler not found for task %q", task.Type()) | ||||
| } | ||||
|  | ||||
| // NotFoundHandler returns a simple task handler that returns a ``not found`` error. | ||||
|   | ||||
| @@ -68,7 +68,7 @@ func TestServeMux(t *testing.T) { | ||||
| 		} | ||||
|  | ||||
| 		if called != tc.want { | ||||
| 			t.Errorf("%q handler was called for task %q, want %q to be called", called, task.Type, tc.want) | ||||
| 			t.Errorf("%q handler was called for task %q, want %q to be called", called, task.Type(), tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -124,7 +124,7 @@ func TestServeMuxNotFound(t *testing.T) { | ||||
| 		task := NewTask(tc.typename, nil) | ||||
| 		err := mux.ProcessTask(context.Background(), task) | ||||
| 		if err == nil { | ||||
| 			t.Errorf("ProcessTask did not return error for task %q, should return 'not found' error", task.Type) | ||||
| 			t.Errorf("ProcessTask did not return error for task %q, should return 'not found' error", task.Type()) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -164,7 +164,7 @@ func TestServeMuxMiddlewares(t *testing.T) { | ||||
| 		} | ||||
|  | ||||
| 		if called != tc.want { | ||||
| 			t.Errorf("%q handler was called for task %q, want %q to be called", called, task.Type, tc.want) | ||||
| 			t.Errorf("%q handler was called for task %q, want %q to be called", called, task.Type(), tc.want) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										139
									
								
								server.go
									
									
									
									
									
								
							
							
						
						
									
										139
									
								
								server.go
									
									
									
									
									
								
							| @@ -15,28 +15,30 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| ) | ||||
|  | ||||
| // Server is responsible for managing the task processing. | ||||
| // Server is responsible for task processing and task lifecycle management. | ||||
| // | ||||
| // Server pulls tasks off queues and processes them. | ||||
| // If the processing of a task is unsuccessful, server will schedule it for a retry. | ||||
| // | ||||
| // A task will be retried until either the task gets processed successfully | ||||
| // or until it reaches its max retry count. | ||||
| // | ||||
| // If a task exhausts its retries, it will be moved to the archive and | ||||
| // will be kept in the archive for some time until a certain condition is met | ||||
| // (e.g., archive size reaches a certain limit, or the task has been in the | ||||
| // archive for a certain amount of time). | ||||
| // will be kept in the archive set. | ||||
| // Note that the archive size is finite and once it reaches its max size, | ||||
| // oldest tasks in the archive will be deleted. | ||||
| type Server struct { | ||||
| 	logger *log.Logger | ||||
|  | ||||
| 	broker base.Broker | ||||
|  | ||||
| 	status *base.ServerStatus | ||||
| 	state *base.ServerState | ||||
|  | ||||
| 	// wait group to wait for all goroutines to finish. | ||||
| 	wg            sync.WaitGroup | ||||
| @@ -47,6 +49,7 @@ type Server struct { | ||||
| 	subscriber    *subscriber | ||||
| 	recoverer     *recoverer | ||||
| 	healthchecker *healthchecker | ||||
| 	janitor       *janitor | ||||
| } | ||||
|  | ||||
| // Config specifies the server's background-task processing behavior. | ||||
| @@ -62,6 +65,14 @@ type Config struct { | ||||
| 	// By default, it uses exponential backoff algorithm to calculate the delay. | ||||
| 	RetryDelayFunc RetryDelayFunc | ||||
|  | ||||
| 	// Predicate function to determine whether the error returned from Handler is a failure. | ||||
| 	// If the function returns false, Server will not increment the retried counter for the task, | ||||
| 	// and Server won't record the queue stats (processed and failed stats) to avoid skewing the error | ||||
| 	// rate of the queue. | ||||
| 	// | ||||
| 	// By default, if the given error is non-nil the function returns true. | ||||
| 	IsFailure func(error) bool | ||||
|  | ||||
| 	// List of queues to process with given priority value. Keys are the names of the | ||||
| 	// queues and values are associated priority value. | ||||
| 	// | ||||
| @@ -266,6 +277,8 @@ func DefaultRetryDelayFunc(n int, e error, t *Task) time.Duration { | ||||
| 	return time.Duration(s) * time.Second | ||||
| } | ||||
|  | ||||
| func defaultIsFailureFunc(err error) bool { return err != nil } | ||||
|  | ||||
| var defaultQueueConfig = map[string]int{ | ||||
| 	base.DefaultQueueName: 1, | ||||
| } | ||||
| @@ -277,8 +290,12 @@ const ( | ||||
| ) | ||||
|  | ||||
| // NewServer returns a new Server given a redis connection option | ||||
| // and background processing configuration. | ||||
| // and server configuration. | ||||
| func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	c, ok := r.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("asynq: unsupported RedisConnOpt type %T", r)) | ||||
| 	} | ||||
| 	n := cfg.Concurrency | ||||
| 	if n < 1 { | ||||
| 		n = runtime.NumCPU() | ||||
| @@ -287,8 +304,15 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	if delayFunc == nil { | ||||
| 		delayFunc = DefaultRetryDelayFunc | ||||
| 	} | ||||
| 	isFailureFunc := cfg.IsFailure | ||||
| 	if isFailureFunc == nil { | ||||
| 		isFailureFunc = defaultIsFailureFunc | ||||
| 	} | ||||
| 	queues := make(map[string]int) | ||||
| 	for qname, p := range cfg.Queues { | ||||
| 		if err := base.ValidateQueueName(qname); err != nil { | ||||
| 			continue // ignore invalid queue names | ||||
| 		} | ||||
| 		if p > 0 { | ||||
| 			queues[qname] = p | ||||
| 		} | ||||
| @@ -315,11 +339,11 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 	} | ||||
| 	logger.SetLevel(toInternalLogLevel(loglevel)) | ||||
|  | ||||
| 	rdb := rdb.NewRDB(createRedisClient(r)) | ||||
| 	starting := make(chan *base.TaskMessage) | ||||
| 	rdb := rdb.NewRDB(c) | ||||
| 	starting := make(chan *workerInfo) | ||||
| 	finished := make(chan *base.TaskMessage) | ||||
| 	syncCh := make(chan *syncRequest) | ||||
| 	status := base.NewServerStatus(base.StatusIdle) | ||||
| 	state := base.NewServerState() | ||||
| 	cancels := base.NewCancelations() | ||||
|  | ||||
| 	syncer := newSyncer(syncerParams{ | ||||
| @@ -334,7 +358,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		concurrency:    n, | ||||
| 		queues:         queues, | ||||
| 		strictPriority: cfg.StrictPriority, | ||||
| 		status:         status, | ||||
| 		state:          state, | ||||
| 		starting:       starting, | ||||
| 		finished:       finished, | ||||
| 	}) | ||||
| @@ -353,6 +377,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		logger:          logger, | ||||
| 		broker:          rdb, | ||||
| 		retryDelayFunc:  delayFunc, | ||||
| 		isFailureFunc:   isFailureFunc, | ||||
| 		syncCh:          syncCh, | ||||
| 		cancelations:    cancels, | ||||
| 		concurrency:     n, | ||||
| @@ -367,6 +392,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		logger:         logger, | ||||
| 		broker:         rdb, | ||||
| 		retryDelayFunc: delayFunc, | ||||
| 		isFailureFunc:  isFailureFunc, | ||||
| 		queues:         qnames, | ||||
| 		interval:       1 * time.Minute, | ||||
| 	}) | ||||
| @@ -376,10 +402,16 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		interval:        healthcheckInterval, | ||||
| 		healthcheckFunc: cfg.HealthCheckFunc, | ||||
| 	}) | ||||
| 	janitor := newJanitor(janitorParams{ | ||||
| 		logger:   logger, | ||||
| 		broker:   rdb, | ||||
| 		queues:   qnames, | ||||
| 		interval: 8 * time.Second, | ||||
| 	}) | ||||
| 	return &Server{ | ||||
| 		logger:        logger, | ||||
| 		broker:        rdb, | ||||
| 		status:        status, | ||||
| 		state:         state, | ||||
| 		forwarder:     forwarder, | ||||
| 		processor:     processor, | ||||
| 		syncer:        syncer, | ||||
| @@ -387,6 +419,7 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| 		subscriber:    subscriber, | ||||
| 		recoverer:     recoverer, | ||||
| 		healthchecker: healthchecker, | ||||
| 		janitor:       janitor, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -395,11 +428,13 @@ func NewServer(r RedisConnOpt, cfg Config) *Server { | ||||
| // ProcessTask should return nil if the processing of a task | ||||
| // is successful. | ||||
| // | ||||
| // If ProcessTask return a non-nil error or panics, the task | ||||
| // will be retried after delay. | ||||
| // One exception to this rule is when ProcessTask returns SkipRetry error. | ||||
| // If the returned error is SkipRetry or the error wraps SkipRetry, retry is | ||||
| // skipped and task will be archived instead. | ||||
| // If ProcessTask returns a non-nil error or panics, the task | ||||
| // will be retried after delay if retry-count is remaining, | ||||
| // otherwise the task will be archived. | ||||
| // | ||||
| // 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. | ||||
| type Handler interface { | ||||
| 	ProcessTask(context.Context, *Task) error | ||||
| } | ||||
| @@ -415,43 +450,46 @@ func (fn HandlerFunc) ProcessTask(ctx context.Context, task *Task) error { | ||||
| 	return fn(ctx, task) | ||||
| } | ||||
|  | ||||
| // ErrServerStopped indicates that the operation is now illegal because of the server being stopped. | ||||
| var ErrServerStopped = errors.New("asynq: the server has been stopped") | ||||
| // ErrServerClosed indicates that the operation is now illegal because of the server has been shutdown. | ||||
| var ErrServerClosed = errors.New("asynq: Server closed") | ||||
|  | ||||
| // Run starts the background-task processing and blocks until | ||||
| // Run starts the task processing and blocks until | ||||
| // an os signal to exit the program is received. Once it receives | ||||
| // a signal, it gracefully shuts down all active workers and other | ||||
| // goroutines to process the tasks. | ||||
| // | ||||
| // Run returns any error encountered during server startup time. | ||||
| // If the server has already been stopped, ErrServerStopped is returned. | ||||
| // Run returns any error encountered at server startup time. | ||||
| // If the server has already been shutdown, ErrServerClosed is returned. | ||||
| func (srv *Server) Run(handler Handler) error { | ||||
| 	if err := srv.Start(handler); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	srv.waitForSignals() | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Start starts the worker server. Once the server has started, | ||||
| // it pulls tasks off queues and starts a worker goroutine for each task. | ||||
| // it pulls tasks off queues and starts a worker goroutine for each task | ||||
| // and then call Handler to process it. | ||||
| // Tasks are processed concurrently by the workers up to the number of | ||||
| // concurrency specified at the initialization time. | ||||
| // concurrency specified in Config.Concurrency. | ||||
| // | ||||
| // Start returns any error encountered during server startup time. | ||||
| // If the server has already been stopped, ErrServerStopped is returned. | ||||
| // Start returns any error encountered at server startup time. | ||||
| // If the server has already been shutdown, ErrServerClosed is returned. | ||||
| func (srv *Server) Start(handler Handler) error { | ||||
| 	if handler == nil { | ||||
| 		return fmt.Errorf("asynq: server cannot run with nil handler") | ||||
| 	} | ||||
| 	switch srv.status.Get() { | ||||
| 	case base.StatusRunning: | ||||
| 	switch srv.state.Get() { | ||||
| 	case base.StateActive: | ||||
| 		return fmt.Errorf("asynq: the server is already running") | ||||
| 	case base.StatusStopped: | ||||
| 		return ErrServerStopped | ||||
| 	case base.StateStopped: | ||||
| 		return fmt.Errorf("asynq: the server is in the stopped state. Waiting for shutdown.") | ||||
| 	case base.StateClosed: | ||||
| 		return ErrServerClosed | ||||
| 	} | ||||
| 	srv.status.Set(base.StatusRunning) | ||||
| 	srv.state.Set(base.StateActive) | ||||
| 	srv.processor.handler = handler | ||||
|  | ||||
| 	srv.logger.Info("Starting processing") | ||||
| @@ -463,46 +501,51 @@ func (srv *Server) Start(handler Handler) error { | ||||
| 	srv.recoverer.start(&srv.wg) | ||||
| 	srv.forwarder.start(&srv.wg) | ||||
| 	srv.processor.start(&srv.wg) | ||||
| 	srv.janitor.start(&srv.wg) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Stop stops the worker server. | ||||
| // Shutdown gracefully shuts down the server. | ||||
| // It gracefully closes all active workers. The server will wait for | ||||
| // active workers to finish processing tasks for duration specified in Config.ShutdownTimeout. | ||||
| // If worker didn't finish processing a task during the timeout, the task will be pushed back to Redis. | ||||
| func (srv *Server) Stop() { | ||||
| 	switch srv.status.Get() { | ||||
| 	case base.StatusIdle, base.StatusStopped: | ||||
| func (srv *Server) Shutdown() { | ||||
| 	switch srv.state.Get() { | ||||
| 	case base.StateNew, base.StateClosed: | ||||
| 		// server is not running, do nothing and return. | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	srv.logger.Info("Starting graceful shutdown") | ||||
| 	// Note: The order of termination is important. | ||||
| 	// Note: The order of shutdown is important. | ||||
| 	// Sender goroutines should be terminated before the receiver goroutines. | ||||
| 	// processor -> syncer (via syncCh) | ||||
| 	// processor -> heartbeater (via starting, finished channels) | ||||
| 	srv.forwarder.terminate() | ||||
| 	srv.processor.terminate() | ||||
| 	srv.recoverer.terminate() | ||||
| 	srv.syncer.terminate() | ||||
| 	srv.subscriber.terminate() | ||||
| 	srv.healthchecker.terminate() | ||||
| 	srv.heartbeater.terminate() | ||||
| 	srv.forwarder.shutdown() | ||||
| 	srv.processor.shutdown() | ||||
| 	srv.recoverer.shutdown() | ||||
| 	srv.syncer.shutdown() | ||||
| 	srv.subscriber.shutdown() | ||||
| 	srv.janitor.shutdown() | ||||
| 	srv.healthchecker.shutdown() | ||||
| 	srv.heartbeater.shutdown() | ||||
|  | ||||
| 	srv.wg.Wait() | ||||
|  | ||||
| 	srv.broker.Close() | ||||
| 	srv.status.Set(base.StatusStopped) | ||||
| 	srv.state.Set(base.StateClosed) | ||||
|  | ||||
| 	srv.logger.Info("Exiting") | ||||
| } | ||||
|  | ||||
| // Quiet signals the server to stop pulling new tasks off queues. | ||||
| // Quiet should be used before stopping the server. | ||||
| func (srv *Server) Quiet() { | ||||
| // Stop signals the server to stop pulling new tasks off queues. | ||||
| // Stop can be used before shutting down the server to ensure that all | ||||
| // currently active tasks are processed before server shutdown. | ||||
| // | ||||
| // Stop does not shutdown the server, make sure to call Shutdown before exit. | ||||
| func (srv *Server) Stop() { | ||||
| 	srv.logger.Info("Stopping processor") | ||||
| 	srv.processor.stop() | ||||
| 	srv.status.Set(base.StatusQuiet) | ||||
| 	srv.state.Set(base.StateStopped) | ||||
| 	srv.logger.Info("Processor stopped") | ||||
| } | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq/internal/asynqtest" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/hibiken/asynq/internal/testbroker" | ||||
| 	"go.uber.org/goleak" | ||||
| @@ -18,7 +19,7 @@ import ( | ||||
|  | ||||
| func TestServer(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/go-redis/redis/v7/internal/pool.(*ConnPool).reaper") | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/go-redis/redis/v8/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNoLeaks(t, ignoreOpt) | ||||
|  | ||||
| 	redisConnOpt := getRedisConnOpt(t) | ||||
| @@ -39,22 +40,22 @@ func TestServer(t *testing.T) { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	_, err = c.Enqueue(NewTask("send_email", map[string]interface{}{"recipient_id": 123})) | ||||
| 	_, err = c.Enqueue(NewTask("send_email", asynqtest.JSON(map[string]interface{}{"recipient_id": 123}))) | ||||
| 	if err != nil { | ||||
| 		t.Errorf("could not enqueue a task: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	_, err = c.Enqueue(NewTask("send_email", map[string]interface{}{"recipient_id": 456}), ProcessIn(1*time.Hour)) | ||||
| 	_, err = c.Enqueue(NewTask("send_email", asynqtest.JSON(map[string]interface{}{"recipient_id": 456})), ProcessIn(1*time.Hour)) | ||||
| 	if err != nil { | ||||
| 		t.Errorf("could not enqueue a task: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestServerRun(t *testing.T) { | ||||
| 	// https://github.com/go-redis/redis/issues/1029 | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/go-redis/redis/v7/internal/pool.(*ConnPool).reaper") | ||||
| 	ignoreOpt := goleak.IgnoreTopFunction("github.com/go-redis/redis/v8/internal/pool.(*ConnPool).reaper") | ||||
| 	defer goleak.VerifyNoLeaks(t, ignoreOpt) | ||||
|  | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| @@ -70,7 +71,7 @@ func TestServerRun(t *testing.T) { | ||||
| 	go func() { | ||||
| 		select { | ||||
| 		case <-time.After(10 * time.Second): | ||||
| 			t.Fatal("server did not stop after receiving TERM signal") | ||||
| 			panic("server did not stop after receiving TERM signal") | ||||
| 		case <-done: | ||||
| 		} | ||||
| 	}() | ||||
| @@ -81,16 +82,16 @@ func TestServerRun(t *testing.T) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestServerErrServerStopped(t *testing.T) { | ||||
| func TestServerErrServerClosed(t *testing.T) { | ||||
| 	srv := NewServer(RedisClientOpt{Addr: ":6379"}, Config{LogLevel: testLogLevel}) | ||||
| 	handler := NewServeMux() | ||||
| 	if err := srv.Start(handler); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| 	err := srv.Start(handler) | ||||
| 	if err != ErrServerStopped { | ||||
| 		t.Errorf("Restarting server: (*Server).Start(handler) = %v, want ErrServerStopped error", err) | ||||
| 	if err != ErrServerClosed { | ||||
| 		t.Errorf("Restarting server: (*Server).Start(handler) = %v, want ErrServerClosed error", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -99,7 +100,7 @@ func TestServerErrNilHandler(t *testing.T) { | ||||
| 	err := srv.Start(nil) | ||||
| 	if err == nil { | ||||
| 		t.Error("Starting server with nil handler: (*Server).Start(nil) did not return error") | ||||
| 		srv.Stop() | ||||
| 		srv.Shutdown() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -113,7 +114,7 @@ func TestServerErrServerRunning(t *testing.T) { | ||||
| 	if err == nil { | ||||
| 		t.Error("Calling (*Server).Start(handler) on already running server did not return error") | ||||
| 	} | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestServerWithRedisDown(t *testing.T) { | ||||
| @@ -145,7 +146,7 @@ func TestServerWithRedisDown(t *testing.T) { | ||||
|  | ||||
| 	time.Sleep(3 * time.Second) | ||||
|  | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestServerWithFlakyBroker(t *testing.T) { | ||||
| @@ -169,8 +170,8 @@ func TestServerWithFlakyBroker(t *testing.T) { | ||||
|  | ||||
| 	h := func(ctx context.Context, task *Task) error { | ||||
| 		// force task retry. | ||||
| 		if task.Type == "bad_task" { | ||||
| 			return fmt.Errorf("could not process %q", task.Type) | ||||
| 		if task.Type() == "bad_task" { | ||||
| 			return fmt.Errorf("could not process %q", task.Type()) | ||||
| 		} | ||||
| 		time.Sleep(2 * time.Second) | ||||
| 		return nil | ||||
| @@ -206,7 +207,7 @@ func TestServerWithFlakyBroker(t *testing.T) { | ||||
|  | ||||
| 	time.Sleep(3 * time.Second) | ||||
|  | ||||
| 	srv.Stop() | ||||
| 	srv.Shutdown() | ||||
| } | ||||
|  | ||||
| func TestLogLevel(t *testing.T) { | ||||
|   | ||||
| @@ -22,7 +22,7 @@ func (srv *Server) waitForSignals() { | ||||
| 	for { | ||||
| 		sig := <-sigs | ||||
| 		if sig == unix.SIGTSTP { | ||||
| 			srv.Quiet() | ||||
| 			srv.Stop() | ||||
| 			continue | ||||
| 		} | ||||
| 		break | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/log" | ||||
| ) | ||||
| @@ -43,7 +43,7 @@ func newSubscriber(params subscriberParams) *subscriber { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *subscriber) terminate() { | ||||
| func (s *subscriber) shutdown() { | ||||
| 	s.logger.Debug("Subscriber shutting down...") | ||||
| 	// Signal the subscriber goroutine to stop. | ||||
| 	s.done <- struct{}{} | ||||
|   | ||||
| @@ -46,7 +46,7 @@ func TestSubscriber(t *testing.T) { | ||||
| 		}) | ||||
| 		var wg sync.WaitGroup | ||||
| 		subscriber.start(&wg) | ||||
| 		defer subscriber.terminate() | ||||
| 		defer subscriber.shutdown() | ||||
|  | ||||
| 		// wait for subscriber to establish connection to pubsub channel | ||||
| 		time.Sleep(time.Second) | ||||
| @@ -91,7 +91,7 @@ func TestSubscriberWithRedisDown(t *testing.T) { | ||||
| 	testBroker.Sleep() // simulate a situation where subscriber cannot connect to redis. | ||||
| 	var wg sync.WaitGroup | ||||
| 	subscriber.start(&wg) | ||||
| 	defer subscriber.terminate() | ||||
| 	defer subscriber.shutdown() | ||||
|  | ||||
| 	time.Sleep(2 * time.Second) // subscriber should wait and retry connecting to redis. | ||||
|  | ||||
|   | ||||
| @@ -46,7 +46,7 @@ func newSyncer(params syncerParams) *syncer { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *syncer) terminate() { | ||||
| func (s *syncer) shutdown() { | ||||
| 	s.logger.Debug("Syncer shutting down...") | ||||
| 	// Signal the syncer goroutine to stop. | ||||
| 	s.done <- struct{}{} | ||||
|   | ||||
| @@ -35,7 +35,7 @@ func TestSyncer(t *testing.T) { | ||||
| 	}) | ||||
| 	var wg sync.WaitGroup | ||||
| 	syncer.start(&wg) | ||||
| 	defer syncer.terminate() | ||||
| 	defer syncer.shutdown() | ||||
|  | ||||
| 	for _, msg := range inProgress { | ||||
| 		m := msg | ||||
| @@ -66,7 +66,7 @@ func TestSyncerRetry(t *testing.T) { | ||||
|  | ||||
| 	var wg sync.WaitGroup | ||||
| 	syncer.start(&wg) | ||||
| 	defer syncer.terminate() | ||||
| 	defer syncer.shutdown() | ||||
|  | ||||
| 	var ( | ||||
| 		mu      sync.Mutex | ||||
| @@ -131,7 +131,7 @@ func TestSyncerDropsStaleRequests(t *testing.T) { | ||||
| 	} | ||||
|  | ||||
| 	time.Sleep(2 * interval) // ensure that syncer runs at least once | ||||
| 	syncer.terminate() | ||||
| 	syncer.shutdown() | ||||
|  | ||||
| 	mu.Lock() | ||||
| 	if n != 0 { | ||||
|   | ||||
| @@ -63,7 +63,7 @@ func cronList(cmd *cobra.Command, args []string) { | ||||
| 	cols := []string{"EntryID", "Spec", "Type", "Payload", "Options", "Next", "Prev"} | ||||
| 	printRows := func(w io.Writer, tmpl string) { | ||||
| 		for _, e := range entries { | ||||
| 			fmt.Fprintf(w, tmpl, e.ID, e.Spec, e.Task.Type, e.Task.Payload, e.Opts, | ||||
| 			fmt.Fprintf(w, tmpl, e.ID, e.Spec, e.Task.Type(), sprintBytes(e.Task.Payload()), e.Opts, | ||||
| 				nextEnqueue(e.Next), prevEnqueue(e.Prev)) | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
| @@ -1,389 +0,0 @@ | ||||
| // Copyright 2020 Kentaro Hibino. All rights reserved. | ||||
| // Use of this source code is governed by a MIT license | ||||
| // that can be found in the LICENSE file. | ||||
|  | ||||
| package cmd | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/spf13/cast" | ||||
| 	"github.com/spf13/cobra" | ||||
| 	"github.com/spf13/viper" | ||||
| ) | ||||
|  | ||||
| var migrateCmd = &cobra.Command{ | ||||
| 	Use:   "migrate", | ||||
| 	Short: fmt.Sprintf("Migrate all tasks to be compatible with asynq v%s", base.Version), | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   migrate, | ||||
| } | ||||
|  | ||||
| func init() { | ||||
| 	rootCmd.AddCommand(migrateCmd) | ||||
| } | ||||
|  | ||||
| func migrate(cmd *cobra.Command, args []string) { | ||||
| 	c := redis.NewClient(&redis.Options{ | ||||
| 		Addr:     viper.GetString("uri"), | ||||
| 		DB:       viper.GetInt("db"), | ||||
| 		Password: viper.GetString("password"), | ||||
| 	}) | ||||
| 	r := createRDB() | ||||
|  | ||||
| 	/*** Migrate from 0.9 to 0.10, 0.11 compatible ***/ | ||||
| 	lists := []string{"asynq:in_progress"} | ||||
| 	allQueues, err := c.SMembers(base.AllQueues).Result() | ||||
| 	if err != nil { | ||||
| 		printError(fmt.Errorf("could not read all queues: %v", err)) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	lists = append(lists, allQueues...) | ||||
| 	for _, key := range lists { | ||||
| 		if err := migrateList(c, key); err != nil { | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	zsets := []string{"asynq:scheduled", "asynq:retry", "asynq:dead"} | ||||
| 	for _, key := range zsets { | ||||
| 		if err := migrateZSet(c, key); err != nil { | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/*** Migrate from 0.11 to 0.12 compatible ***/ | ||||
| 	if err := createBackup(c, base.AllQueues); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	for _, qkey := range allQueues { | ||||
| 		qname := strings.TrimPrefix(qkey, "asynq:queues:") | ||||
| 		if err := c.SAdd(base.AllQueues, qname).Err(); err != nil { | ||||
| 			err = fmt.Errorf("could not add queue name %q to %q set: %v\n", | ||||
| 				qname, base.AllQueues, err) | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
| 	if err := deleteBackup(c, base.AllQueues); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	for _, qkey := range allQueues { | ||||
| 		qname := strings.TrimPrefix(qkey, "asynq:queues:") | ||||
| 		if exists := c.Exists(qkey).Val(); exists == 1 { | ||||
| 			if err := c.Rename(qkey, base.QueueKey(qname)).Err(); err != nil { | ||||
| 				printError(fmt.Errorf("could not rename key %q: %v\n", qkey, err)) | ||||
| 				os.Exit(1) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if err := partitionZSetMembersByQueue(c, "asynq:scheduled", base.ScheduledKey); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	if err := partitionZSetMembersByQueue(c, "asynq:retry", base.RetryKey); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	// Note: base.DeadKey function was renamed in v0.14. We define the legacy function here since we need it for this migration script. | ||||
| 	deadKeyFunc := func(qname string) string { return fmt.Sprintf("asynq:{%s}:dead", qname) }  | ||||
| 	if err := partitionZSetMembersByQueue(c, "asynq:dead", deadKeyFunc); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	if err := partitionZSetMembersByQueue(c, "asynq:deadlines", base.DeadlinesKey); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	if err := partitionListMembersByQueue(c, "asynq:in_progress", base.ActiveKey); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	paused, err := c.SMembers("asynq:paused").Result() | ||||
| 	if err != nil { | ||||
| 		printError(fmt.Errorf("command SMEMBERS asynq:paused failed: %v", err)) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	for _, qkey := range paused { | ||||
| 		qname := strings.TrimPrefix(qkey, "asynq:queues:") | ||||
| 		if err := r.Pause(qname); err != nil { | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
| 	if err := deleteKey(c, "asynq:paused"); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	if err := deleteKey(c, "asynq:servers"); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	if err := deleteKey(c, "asynq:workers"); err != nil { | ||||
| 		printError(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	/*** Migrate from 0.13 to 0.14 compatible ***/ | ||||
|  | ||||
| 	// Move all dead tasks to archived ZSET. | ||||
| 	for _, qname := range allQueues { | ||||
| 		zs, err := c.ZRangeWithScores(deadKeyFunc(qname), 0, -1).Result() | ||||
| 		if err != nil { | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 		for _, z := range zs { | ||||
| 			if err := c.ZAdd(base.ArchivedKey(qname), &z).Err(); err != nil { | ||||
| 				printError(err) | ||||
| 				os.Exit(1) | ||||
| 			} | ||||
| 		} | ||||
| 		if err := deleteKey(c, deadKeyFunc(qname)); err != nil { | ||||
| 			printError(err) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func backupKey(key string) string { | ||||
| 	return fmt.Sprintf("%s:backup", key) | ||||
| } | ||||
|  | ||||
| func createBackup(c *redis.Client, key string) error { | ||||
| 	err := c.Rename(key, backupKey(key)).Err() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("could not rename key %q: %v", key, err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func deleteBackup(c *redis.Client, key string) error { | ||||
| 	return deleteKey(c, backupKey(key)) | ||||
| } | ||||
|  | ||||
| func deleteKey(c *redis.Client, key string) error { | ||||
| 	exists := c.Exists(key).Val() | ||||
| 	if exists == 0 { | ||||
| 		// key does not exist | ||||
| 		return nil | ||||
| 	} | ||||
| 	err := c.Del(key).Err() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("could not delete key %q: %v", key, err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func printError(err error) { | ||||
| 	fmt.Println(err) | ||||
| 	fmt.Println() | ||||
| 	fmt.Println("Migrate command error") | ||||
| 	fmt.Println("Please file an issue on Github at https://github.com/hibiken/asynq/issues/new/choose") | ||||
| } | ||||
|  | ||||
| func partitionZSetMembersByQueue(c *redis.Client, key string, newKeyFunc func(string) string) error { | ||||
| 	zs, err := c.ZRangeWithScores(key, 0, -1).Result() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("command ZRANGE %s 0 -1 WITHSCORES failed: %v", key, err) | ||||
| 	} | ||||
| 	for _, z := range zs { | ||||
| 		s := cast.ToString(z.Member) | ||||
| 		msg, err := base.DecodeMessage(s) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("could not decode message from %q: %v", key, err) | ||||
| 		} | ||||
| 		if err := c.ZAdd(newKeyFunc(msg.Queue), &z).Err(); err != nil { | ||||
| 			return fmt.Errorf("could not add %v to %q: %v", z, newKeyFunc(msg.Queue)) | ||||
| 		} | ||||
| 	} | ||||
| 	if err := deleteKey(c, key); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func partitionListMembersByQueue(c *redis.Client, key string, newKeyFunc func(string) string) error { | ||||
| 	data, err := c.LRange(key, 0, -1).Result() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("command LRANGE %s 0 -1 failed: %v", key, err) | ||||
| 	} | ||||
| 	for _, s := range data { | ||||
| 		msg, err := base.DecodeMessage(s) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("could not decode message from %q: %v", key, err) | ||||
| 		} | ||||
| 		if err := c.LPush(newKeyFunc(msg.Queue), s).Err(); err != nil { | ||||
| 			return fmt.Errorf("could not add %v to %q: %v", s, newKeyFunc(msg.Queue)) | ||||
| 		} | ||||
| 	} | ||||
| 	if err := deleteKey(c, key); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| type oldTaskMessage struct { | ||||
| 	// Unchanged | ||||
| 	Type      string | ||||
| 	Payload   map[string]interface{} | ||||
| 	ID        uuid.UUID | ||||
| 	Queue     string | ||||
| 	Retry     int | ||||
| 	Retried   int | ||||
| 	ErrorMsg  string | ||||
| 	UniqueKey string | ||||
|  | ||||
| 	// Following fields have changed. | ||||
|  | ||||
| 	// Deadline specifies the deadline for the task. | ||||
| 	// Task won't be processed if it exceeded its deadline. | ||||
| 	// The string shoulbe be in RFC3339 format. | ||||
| 	// | ||||
| 	// time.Time's zero value means no deadline. | ||||
| 	Timeout string | ||||
|  | ||||
| 	// Deadline specifies the deadline for the task. | ||||
| 	// Task won't be processed if it exceeded its deadline. | ||||
| 	// The string shoulbe be in RFC3339 format. | ||||
| 	// | ||||
| 	// time.Time's zero value means no deadline. | ||||
| 	Deadline string | ||||
| } | ||||
|  | ||||
| var defaultTimeout = 30 * time.Minute | ||||
|  | ||||
| func convertMessage(old *oldTaskMessage) (*base.TaskMessage, error) { | ||||
| 	timeout, err := time.ParseDuration(old.Timeout) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("could not parse Timeout field of %+v", old) | ||||
| 	} | ||||
| 	deadline, err := time.Parse(time.RFC3339, old.Deadline) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("could not parse Deadline field of %+v", old) | ||||
| 	} | ||||
| 	if timeout == 0 && deadline.IsZero() { | ||||
| 		timeout = defaultTimeout | ||||
| 	} | ||||
| 	if deadline.IsZero() { | ||||
| 		// Zero value used to be time.Time{}, | ||||
| 		// in the new schema zero value is represented by | ||||
| 		// zero in Unix time. | ||||
| 		deadline = time.Unix(0, 0) | ||||
| 	} | ||||
| 	return &base.TaskMessage{ | ||||
| 		Type:      old.Type, | ||||
| 		Payload:   old.Payload, | ||||
| 		ID:        uuid.New(), | ||||
| 		Queue:     old.Queue, | ||||
| 		Retry:     old.Retry, | ||||
| 		Retried:   old.Retried, | ||||
| 		ErrorMsg:  old.ErrorMsg, | ||||
| 		UniqueKey: old.UniqueKey, | ||||
| 		Timeout:   int64(timeout.Seconds()), | ||||
| 		Deadline:  deadline.Unix(), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| func deserialize(s string) (*base.TaskMessage, error) { | ||||
| 	// Try deserializing as old message. | ||||
| 	d := json.NewDecoder(strings.NewReader(s)) | ||||
| 	d.UseNumber() | ||||
| 	var old *oldTaskMessage | ||||
| 	if err := d.Decode(&old); err != nil { | ||||
| 		// Try deserializing as new message. | ||||
| 		d = json.NewDecoder(strings.NewReader(s)) | ||||
| 		d.UseNumber() | ||||
| 		var msg *base.TaskMessage | ||||
| 		if err := d.Decode(&msg); err != nil { | ||||
| 			return nil, fmt.Errorf("could not deserialize %s into task message: %v", s, err) | ||||
| 		} | ||||
| 		return msg, nil | ||||
| 	} | ||||
| 	return convertMessage(old) | ||||
| } | ||||
|  | ||||
| func migrateZSet(c *redis.Client, key string) error { | ||||
| 	if c.Exists(key).Val() == 0 { | ||||
| 		// skip if key doesn't exist. | ||||
| 		return nil | ||||
| 	} | ||||
| 	res, err := c.ZRangeWithScores(key, 0, -1).Result() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	var msgs []*redis.Z | ||||
| 	for _, z := range res { | ||||
| 		s, err := cast.ToStringE(z.Member) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("could not cast to string: %v", err) | ||||
| 		} | ||||
| 		msg, err := deserialize(s) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		encoded, err := base.EncodeMessage(msg) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("could not encode message from %q: %v", key, err) | ||||
| 		} | ||||
| 		msgs = append(msgs, &redis.Z{Score: z.Score, Member: encoded}) | ||||
| 	} | ||||
| 	if err := c.Rename(key, key+":backup").Err(); err != nil { | ||||
| 		return fmt.Errorf("could not rename key %q: %v", key, err) | ||||
| 	} | ||||
| 	if err := c.ZAdd(key, msgs...).Err(); err != nil { | ||||
| 		return fmt.Errorf("could not write new messages to %q: %v", key, err) | ||||
| 	} | ||||
| 	if err := c.Del(key + ":backup").Err(); err != nil { | ||||
| 		return fmt.Errorf("could not delete back up key %q: %v", key+":backup", err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func migrateList(c *redis.Client, key string) error { | ||||
| 	if c.Exists(key).Val() == 0 { | ||||
| 		// skip if key doesn't exist. | ||||
| 		return nil | ||||
| 	} | ||||
| 	res, err := c.LRange(key, 0, -1).Result() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	var msgs []interface{} | ||||
| 	for _, s := range res { | ||||
| 		msg, err := deserialize(s) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		encoded, err := base.EncodeMessage(msg) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("could not encode message from %q: %v", key, err) | ||||
| 		} | ||||
| 		msgs = append(msgs, encoded) | ||||
| 	} | ||||
| 	if err := c.Rename(key, key+":backup").Err(); err != nil { | ||||
| 		return fmt.Errorf("could not rename key %q: %v", key, err) | ||||
| 	} | ||||
| 	if err := c.LPush(key, msgs...).Err(); err != nil { | ||||
| 		return fmt.Errorf("could not write new messages to %q: %v", key, err) | ||||
| 	} | ||||
| 	if err := c.Del(key + ":backup").Err(); err != nil { | ||||
| 		return fmt.Errorf("could not delete back up key %q: %v", key+":backup", err) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| @@ -11,7 +11,7 @@ import ( | ||||
|  | ||||
| 	"github.com/fatih/color" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| 	"github.com/hibiken/asynq/internal/errors" | ||||
| 	"github.com/spf13/cobra" | ||||
| ) | ||||
|  | ||||
| @@ -82,7 +82,7 @@ func queueList(cmd *cobra.Command, args []string) { | ||||
| 	type queueInfo struct { | ||||
| 		name    string | ||||
| 		keyslot int64 | ||||
| 		nodes   []asynq.ClusterNode | ||||
| 		nodes   []*asynq.ClusterNode | ||||
| 	} | ||||
| 	inspector := createInspector() | ||||
| 	queues, err := inspector.Queues() | ||||
| @@ -90,7 +90,7 @@ func queueList(cmd *cobra.Command, args []string) { | ||||
| 		fmt.Printf("error: Could not fetch list of queues: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	var qs []queueInfo | ||||
| 	var qs []*queueInfo | ||||
| 	for _, qname := range queues { | ||||
| 		q := queueInfo{name: qname} | ||||
| 		if useRedisCluster { | ||||
| @@ -107,7 +107,7 @@ func queueList(cmd *cobra.Command, args []string) { | ||||
| 			} | ||||
| 			q.nodes = nodes | ||||
| 		} | ||||
| 		qs = append(qs, q) | ||||
| 		qs = append(qs, &q) | ||||
| 	} | ||||
| 	if useRedisCluster { | ||||
| 		printTable( | ||||
| @@ -129,43 +129,42 @@ func queueInspect(cmd *cobra.Command, args []string) { | ||||
| 	inspector := createInspector() | ||||
| 	for i, qname := range args { | ||||
| 		if i > 0 { | ||||
| 			fmt.Printf("\n%s\n", separator) | ||||
| 			fmt.Printf("\n%s\n\n", separator) | ||||
| 		} | ||||
| 		fmt.Println() | ||||
| 		stats, err := inspector.CurrentStats(qname) | ||||
| 		info, err := inspector.GetQueueInfo(qname) | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		printQueueStats(stats) | ||||
| 		printQueueInfo(info) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func printQueueStats(s *asynq.QueueStats) { | ||||
| func printQueueInfo(info *asynq.QueueInfo) { | ||||
| 	bold := color.New(color.Bold) | ||||
| 	bold.Println("Queue Info") | ||||
| 	fmt.Printf("Name:   %s\n", s.Queue) | ||||
| 	fmt.Printf("Size:   %d\n", s.Size) | ||||
| 	fmt.Printf("Paused: %t\n\n", s.Paused) | ||||
| 	fmt.Printf("Name:   %s\n", info.Queue) | ||||
| 	fmt.Printf("Size:   %d\n", info.Size) | ||||
| 	fmt.Printf("Paused: %t\n\n", info.Paused) | ||||
| 	bold.Println("Task Count by State") | ||||
| 	printTable( | ||||
| 		[]string{"active", "pending", "scheduled", "retry", "archived"}, | ||||
| 		[]string{"active", "pending", "scheduled", "retry", "archived", "completed"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			fmt.Fprintf(w, tmpl, s.Active, s.Pending, s.Scheduled, s.Retry, s.Archived) | ||||
| 			fmt.Fprintf(w, tmpl, info.Active, info.Pending, info.Scheduled, info.Retry, info.Archived, info.Completed) | ||||
| 		}, | ||||
| 	) | ||||
| 	fmt.Println() | ||||
| 	bold.Printf("Daily Stats %s UTC\n", s.Timestamp.UTC().Format("2006-01-02")) | ||||
| 	bold.Printf("Daily Stats %s UTC\n", info.Timestamp.UTC().Format("2006-01-02")) | ||||
| 	printTable( | ||||
| 		[]string{"processed", "failed", "error rate"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			var errRate string | ||||
| 			if s.Processed == 0 { | ||||
| 			if info.Processed == 0 { | ||||
| 				errRate = "N/A" | ||||
| 			} else { | ||||
| 				errRate = fmt.Sprintf("%.2f%%", float64(s.Failed)/float64(s.Processed)*100) | ||||
| 				errRate = fmt.Sprintf("%.2f%%", float64(info.Failed)/float64(info.Processed)*100) | ||||
| 			} | ||||
| 			fmt.Fprintf(w, tmpl, s.Processed, s.Failed, errRate) | ||||
| 			fmt.Fprintf(w, tmpl, info.Processed, info.Failed, errRate) | ||||
| 		}, | ||||
| 	) | ||||
| } | ||||
| @@ -179,9 +178,9 @@ func queueHistory(cmd *cobra.Command, args []string) { | ||||
| 	inspector := createInspector() | ||||
| 	for i, qname := range args { | ||||
| 		if i > 0 { | ||||
| 			fmt.Printf("\n%s\n", separator) | ||||
| 			fmt.Printf("\n%s\n\n", separator) | ||||
| 		} | ||||
| 		fmt.Printf("\nQueue: %s\n\n", qname) | ||||
| 		fmt.Printf("Queue: %s\n\n", qname) | ||||
| 		stats, err := inspector.History(qname, days) | ||||
| 		if err != nil { | ||||
| 			fmt.Printf("error: %v\n", err) | ||||
| @@ -244,7 +243,7 @@ func queueRemove(cmd *cobra.Command, args []string) { | ||||
| 	for _, qname := range args { | ||||
| 		err = r.RemoveQueue(qname, force) | ||||
| 		if err != nil { | ||||
| 			if _, ok := err.(*rdb.ErrQueueNotEmpty); ok { | ||||
| 			if errors.IsQueueNotEmpty(err) { | ||||
| 				fmt.Printf("error: %v\nIf you are sure you want to delete it, run 'asynq queue rm --force %s'\n", err, qname) | ||||
| 				continue | ||||
| 			} | ||||
|   | ||||
| @@ -11,8 +11,10 @@ import ( | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 	"text/tabwriter" | ||||
| 	"unicode" | ||||
| 	"unicode/utf8" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v7" | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/internal/base" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| @@ -136,23 +138,24 @@ func createRDB() *rdb.RDB { | ||||
|  | ||||
| // createRDB creates a Inspector instance using flag values and returns it. | ||||
| func createInspector() *asynq.Inspector { | ||||
| 	var connOpt asynq.RedisConnOpt | ||||
| 	return asynq.NewInspector(getRedisConnOpt()) | ||||
| } | ||||
|  | ||||
| func getRedisConnOpt() asynq.RedisConnOpt { | ||||
| 	if useRedisCluster { | ||||
| 		addrs := strings.Split(viper.GetString("cluster_addrs"), ",") | ||||
| 		connOpt = asynq.RedisClusterClientOpt{ | ||||
| 		return asynq.RedisClusterClientOpt{ | ||||
| 			Addrs:     addrs, | ||||
| 			Password:  viper.GetString("password"), | ||||
| 			TLSConfig: getTLSConfig(), | ||||
| 		} | ||||
| 	} else { | ||||
| 		connOpt = asynq.RedisClientOpt{ | ||||
| 	} | ||||
| 	return asynq.RedisClientOpt{ | ||||
| 		Addr:      viper.GetString("uri"), | ||||
| 		DB:        viper.GetInt("db"), | ||||
| 		Password:  viper.GetString("password"), | ||||
| 		TLSConfig: getTLSConfig(), | ||||
| 	} | ||||
| 	} | ||||
| 	return asynq.NewInspector(connOpt) | ||||
| } | ||||
|  | ||||
| func getTLSConfig() *tls.Config { | ||||
| @@ -195,3 +198,28 @@ func printTable(cols []string, printRows func(w io.Writer, tmpl string)) { | ||||
| 	printRows(tw, format) | ||||
| 	tw.Flush() | ||||
| } | ||||
|  | ||||
| // sprintBytes returns a string representation of the given byte slice if data is printable. | ||||
| // If data is not printable, it returns a string describing it is not printable. | ||||
| func sprintBytes(payload []byte) string { | ||||
| 	if !isPrintable(payload) { | ||||
| 		return "non-printable bytes" | ||||
| 	} | ||||
| 	return string(payload) | ||||
| } | ||||
|  | ||||
| func isPrintable(data []byte) bool { | ||||
| 	if !utf8.Valid(data) { | ||||
| 		return false | ||||
| 	} | ||||
| 	isAllSpace := true | ||||
| 	for _, r := range string(data) { | ||||
| 		if !unicode.IsPrint(r) { | ||||
| 			return false | ||||
| 		} | ||||
| 		if !unicode.IsSpace(r) { | ||||
| 			isAllSpace = false | ||||
| 		} | ||||
| 	} | ||||
| 	return !isAllSpace | ||||
| } | ||||
|   | ||||
| @@ -35,11 +35,11 @@ The command shows the following for each server: | ||||
| * Host and PID of the process in which the server is running | ||||
| * Number of active workers out of worker pool | ||||
| * Queue configuration | ||||
| * State of the worker server ("running" | "quiet") | ||||
| * State of the worker server ("active" | "stopped") | ||||
| * Time the server was started | ||||
|  | ||||
| A "running" server is pulling tasks from queues and processing them. | ||||
| A "quiet" server is no longer pulling new tasks from queues`, | ||||
| A "active" server is pulling tasks from queues and processing them. | ||||
| A "stopped" server is no longer pulling new tasks from queues`, | ||||
| 	Run: serverList, | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -7,11 +7,13 @@ package cmd | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"math" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"text/tabwriter" | ||||
| 	"time" | ||||
| 	"unicode/utf8" | ||||
|  | ||||
| 	"github.com/fatih/color" | ||||
| 	"github.com/hibiken/asynq/internal/rdb" | ||||
| @@ -22,7 +24,7 @@ import ( | ||||
| var statsCmd = &cobra.Command{ | ||||
| 	Use:   "stats", | ||||
| 	Short: "Shows current state of the tasks and queues", | ||||
| 	Long: `Stats (aysnqmon stats) will show the overview of tasks and queues at that instant. | ||||
| 	Long: `Stats (aysnq stats) will show the overview of tasks and queues at that instant. | ||||
|  | ||||
| Specifically, the command shows the following: | ||||
| * Number of tasks in each state | ||||
| @@ -58,6 +60,7 @@ type AggregateStats struct { | ||||
| 	Scheduled int | ||||
| 	Retry     int | ||||
| 	Archived  int | ||||
| 	Completed int | ||||
| 	Processed int | ||||
| 	Failed    int | ||||
| 	Timestamp time.Time | ||||
| @@ -85,6 +88,7 @@ func stats(cmd *cobra.Command, args []string) { | ||||
| 		aggStats.Scheduled += s.Scheduled | ||||
| 		aggStats.Retry += s.Retry | ||||
| 		aggStats.Archived += s.Archived | ||||
| 		aggStats.Completed += s.Completed | ||||
| 		aggStats.Processed += s.Processed | ||||
| 		aggStats.Failed += s.Failed | ||||
| 		aggStats.Timestamp = s.Timestamp | ||||
| @@ -124,22 +128,50 @@ func stats(cmd *cobra.Command, args []string) { | ||||
| } | ||||
|  | ||||
| func printStatsByState(s *AggregateStats) { | ||||
| 	format := strings.Repeat("%v\t", 5) + "\n" | ||||
| 	format := strings.Repeat("%v\t", 6) + "\n" | ||||
| 	tw := new(tabwriter.Writer).Init(os.Stdout, 0, 8, 2, ' ', 0) | ||||
| 	fmt.Fprintf(tw, format, "active", "pending", "scheduled", "retry", "archived") | ||||
| 	fmt.Fprintf(tw, format, "----------", "--------", "---------", "-----", "----") | ||||
| 	fmt.Fprintf(tw, format, s.Active, s.Pending, s.Scheduled, s.Retry, s.Archived) | ||||
| 	fmt.Fprintf(tw, format, "active", "pending", "scheduled", "retry", "archived", "completed") | ||||
| 	width := maxInt(9 /* defaultWidth */, maxWidthOf(s.Active, s.Pending, s.Scheduled, s.Retry, s.Archived, s.Completed)) // length of widest column | ||||
| 	sep := strings.Repeat("-", width) | ||||
| 	fmt.Fprintf(tw, format, sep, sep, sep, sep, sep, sep) | ||||
| 	fmt.Fprintf(tw, format, s.Active, s.Pending, s.Scheduled, s.Retry, s.Archived, s.Completed) | ||||
| 	tw.Flush() | ||||
| } | ||||
|  | ||||
| // numDigits returns the number of digits in n. | ||||
| func numDigits(n int) int { | ||||
| 	return len(strconv.Itoa(n)) | ||||
| } | ||||
|  | ||||
| // maxWidthOf returns the max number of digits amount the provided vals. | ||||
| func maxWidthOf(vals ...int) int { | ||||
| 	max := 0 | ||||
| 	for _, v := range vals { | ||||
| 		if vw := numDigits(v); vw > max { | ||||
| 			max = vw | ||||
| 		} | ||||
| 	} | ||||
| 	return max | ||||
| } | ||||
|  | ||||
| func maxInt(a, b int) int { | ||||
| 	return int(math.Max(float64(a), float64(b))) | ||||
| } | ||||
|  | ||||
| func printStatsByQueue(stats []*rdb.Stats) { | ||||
| 	var headers, seps, counts []string | ||||
| 	maxHeaderWidth := 0 | ||||
| 	for _, s := range stats { | ||||
| 		title := queueTitle(s) | ||||
| 		headers = append(headers, title) | ||||
| 		seps = append(seps, strings.Repeat("-", len(title))) | ||||
| 		if w := utf8.RuneCountInString(title); w > maxHeaderWidth { | ||||
| 			maxHeaderWidth = w | ||||
| 		} | ||||
| 		counts = append(counts, strconv.Itoa(s.Size)) | ||||
| 	} | ||||
| 	for i := 0; i < len(headers); i++ { | ||||
| 		seps = append(seps, strings.Repeat("-", maxHeaderWidth)) | ||||
| 	} | ||||
| 	format := strings.Repeat("%v\t", len(headers)) + "\n" | ||||
| 	tw := new(tabwriter.Writer).Init(os.Stdout, 0, 8, 2, ' ', 0) | ||||
| 	fmt.Fprintf(tw, format, toInterfaceSlice(headers)...) | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import ( | ||||
| 	"os" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/fatih/color" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/spf13/cobra" | ||||
| ) | ||||
| @@ -26,23 +27,29 @@ func init() { | ||||
|  | ||||
| 	taskCmd.AddCommand(taskCancelCmd) | ||||
|  | ||||
| 	taskCmd.AddCommand(taskInspectCmd) | ||||
| 	taskInspectCmd.Flags().StringP("queue", "q", "", "queue to which the task belongs") | ||||
| 	taskInspectCmd.Flags().StringP("id", "i", "", "id of the task") | ||||
| 	taskInspectCmd.MarkFlagRequired("queue") | ||||
| 	taskInspectCmd.MarkFlagRequired("id") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskArchiveCmd) | ||||
| 	taskArchiveCmd.Flags().StringP("queue", "q", "", "queue to which the task belongs") | ||||
| 	taskArchiveCmd.Flags().StringP("key", "k", "", "key of the task") | ||||
| 	taskArchiveCmd.Flags().StringP("id", "i", "", "id of the task") | ||||
| 	taskArchiveCmd.MarkFlagRequired("queue") | ||||
| 	taskArchiveCmd.MarkFlagRequired("key") | ||||
| 	taskArchiveCmd.MarkFlagRequired("id") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskDeleteCmd) | ||||
| 	taskDeleteCmd.Flags().StringP("queue", "q", "", "queue to which the task belongs") | ||||
| 	taskDeleteCmd.Flags().StringP("key", "k", "", "key of the task") | ||||
| 	taskDeleteCmd.Flags().StringP("id", "i", "", "id of the task") | ||||
| 	taskDeleteCmd.MarkFlagRequired("queue") | ||||
| 	taskDeleteCmd.MarkFlagRequired("key") | ||||
| 	taskDeleteCmd.MarkFlagRequired("id") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskRunCmd) | ||||
| 	taskRunCmd.Flags().StringP("queue", "q", "", "queue to which the task belongs") | ||||
| 	taskRunCmd.Flags().StringP("key", "k", "", "key of the task") | ||||
| 	taskRunCmd.Flags().StringP("id", "i", "", "id of the task") | ||||
| 	taskRunCmd.MarkFlagRequired("queue") | ||||
| 	taskRunCmd.MarkFlagRequired("key") | ||||
| 	taskRunCmd.MarkFlagRequired("id") | ||||
|  | ||||
| 	taskCmd.AddCommand(taskArchiveAllCmd) | ||||
| 	taskArchiveAllCmd.Flags().StringP("queue", "q", "", "queue to which the tasks belong") | ||||
| @@ -79,6 +86,7 @@ The value for the state flag should be one of: | ||||
| - scheduled | ||||
| - retry | ||||
| - archived | ||||
| - completed | ||||
|  | ||||
| List opeartion paginates the result set. | ||||
| By default, the command fetches the first 30 tasks. | ||||
| @@ -93,6 +101,13 @@ To list the tasks from the second page, run | ||||
| 	Run: taskList, | ||||
| } | ||||
|  | ||||
| var taskInspectCmd = &cobra.Command{ | ||||
| 	Use:   "inspect --queue=QUEUE --id=TASK_ID", | ||||
| 	Short: "Display detailed information on the specified task", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskInspect, | ||||
| } | ||||
|  | ||||
| var taskCancelCmd = &cobra.Command{ | ||||
| 	Use:   "cancel TASK_ID [TASK_ID...]", | ||||
| 	Short: "Cancel one or more active tasks", | ||||
| @@ -101,42 +116,42 @@ var taskCancelCmd = &cobra.Command{ | ||||
| } | ||||
|  | ||||
| var taskArchiveCmd = &cobra.Command{ | ||||
| 	Use:   "archive --queue=QUEUE --key=KEY", | ||||
| 	Short: "Archive a task with the given key", | ||||
| 	Use:   "archive --queue=QUEUE --id=TASK_ID", | ||||
| 	Short: "Archive a task with the given id", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskKill, | ||||
| 	Run:   taskArchive, | ||||
| } | ||||
|  | ||||
| var taskDeleteCmd = &cobra.Command{ | ||||
| 	Use:   "delete --queue=QUEUE --key=KEY", | ||||
| 	Short: "Delete a task with the given key", | ||||
| 	Use:   "delete --queue=QUEUE --id=TASK_ID", | ||||
| 	Short: "Delete a task with the given id", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskDelete, | ||||
| } | ||||
|  | ||||
| var taskRunCmd = &cobra.Command{ | ||||
| 	Use:   "run --queue=QUEUE --key=KEY", | ||||
| 	Short: "Run a task with the given key", | ||||
| 	Use:   "run --queue=QUEUE --id=TASK_ID", | ||||
| 	Short: "Run a task with the given id", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskRun, | ||||
| } | ||||
|  | ||||
| var taskArchiveAllCmd = &cobra.Command{ | ||||
| 	Use:   "archive-all --queue=QUEUE --state=STATE", | ||||
| 	Use:   "archiveall --queue=QUEUE --state=STATE", | ||||
| 	Short: "Archive all tasks in the given state", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskArchiveAll, | ||||
| } | ||||
|  | ||||
| var taskDeleteAllCmd = &cobra.Command{ | ||||
| 	Use:   "delete-all --queue=QUEUE --key=KEY", | ||||
| 	Use:   "deleteall --queue=QUEUE --state=STATE", | ||||
| 	Short: "Delete all tasks in the given state", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskDeleteAll, | ||||
| } | ||||
|  | ||||
| var taskRunAllCmd = &cobra.Command{ | ||||
| 	Use:   "run-all --queue=QUEUE --key=KEY", | ||||
| 	Use:   "runall --queue=QUEUE --state=STATE", | ||||
| 	Short: "Run all tasks in the given state", | ||||
| 	Args:  cobra.NoArgs, | ||||
| 	Run:   taskRunAll, | ||||
| @@ -175,6 +190,8 @@ func taskList(cmd *cobra.Command, args []string) { | ||||
| 		listRetryTasks(qname, pageNum, pageSize) | ||||
| 	case "archived": | ||||
| 		listArchivedTasks(qname, pageNum, pageSize) | ||||
| 	case "completed": | ||||
| 		listCompletedTasks(qname, pageNum, pageSize) | ||||
| 	default: | ||||
| 		fmt.Printf("error: state=%q is not supported\n", state) | ||||
| 		os.Exit(1) | ||||
| @@ -196,7 +213,7 @@ func listActiveTasks(qname string, pageNum, pageSize int) { | ||||
| 		[]string{"ID", "Type", "Payload"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, t.Payload) | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload)) | ||||
| 			} | ||||
| 		}, | ||||
| 	) | ||||
| @@ -217,7 +234,7 @@ func listPendingTasks(qname string, pageNum, pageSize int) { | ||||
| 		[]string{"ID", "Type", "Payload"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, t.Payload) | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload)) | ||||
| 			} | ||||
| 		}, | ||||
| 	) | ||||
| @@ -235,17 +252,26 @@ func listScheduledTasks(qname string, pageNum, pageSize int) { | ||||
| 		return | ||||
| 	} | ||||
| 	printTable( | ||||
| 		[]string{"Key", "Type", "Payload", "Process In"}, | ||||
| 		[]string{"ID", "Type", "Payload", "Process In"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				processIn := fmt.Sprintf("%.0f seconds", | ||||
| 					t.NextProcessAt.Sub(time.Now()).Seconds()) | ||||
| 				fmt.Fprintf(w, tmpl, t.Key(), t.Type, t.Payload, processIn) | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload), formatProcessAt(t.NextProcessAt)) | ||||
| 			} | ||||
| 		}, | ||||
| 	) | ||||
| } | ||||
|  | ||||
| // formatProcessAt formats next process at time to human friendly string. | ||||
| // If processAt time is in the past, returns "right now". | ||||
| // If processAt time is in the future, returns "in xxx" where xxx is the duration from now. | ||||
| func formatProcessAt(processAt time.Time) string { | ||||
| 	d := processAt.Sub(time.Now()) | ||||
| 	if d < 0 { | ||||
| 		return "right now" | ||||
| 	} | ||||
| 	return fmt.Sprintf("in %v", d.Round(time.Second)) | ||||
| } | ||||
|  | ||||
| func listRetryTasks(qname string, pageNum, pageSize int) { | ||||
| 	i := createInspector() | ||||
| 	tasks, err := i.ListRetryTasks(qname, asynq.PageSize(pageSize), asynq.Page(pageNum)) | ||||
| @@ -258,16 +284,11 @@ func listRetryTasks(qname string, pageNum, pageSize int) { | ||||
| 		return | ||||
| 	} | ||||
| 	printTable( | ||||
| 		[]string{"Key", "Type", "Payload", "Next Retry", "Last Error", "Retried", "Max Retry"}, | ||||
| 		[]string{"ID", "Type", "Payload", "Next Retry", "Last Error", "Last Failed", "Retried", "Max Retry"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				var nextRetry string | ||||
| 				if d := t.NextProcessAt.Sub(time.Now()); d > 0 { | ||||
| 					nextRetry = fmt.Sprintf("in %v", d.Round(time.Second)) | ||||
| 				} else { | ||||
| 					nextRetry = "right now" | ||||
| 				} | ||||
| 				fmt.Fprintf(w, tmpl, t.Key(), t.Type, t.Payload, nextRetry, t.ErrorMsg, t.Retried, t.MaxRetry) | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload), formatProcessAt(t.NextProcessAt), | ||||
| 					t.LastErr, formatPastTime(t.LastFailedAt), t.Retried, t.MaxRetry) | ||||
| 			} | ||||
| 		}, | ||||
| 	) | ||||
| @@ -285,19 +306,38 @@ func listArchivedTasks(qname string, pageNum, pageSize int) { | ||||
| 		return | ||||
| 	} | ||||
| 	printTable( | ||||
| 		[]string{"Key", "Type", "Payload", "Last Failed", "Last Error"}, | ||||
| 		[]string{"ID", "Type", "Payload", "Last Failed", "Last Error"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				fmt.Fprintf(w, tmpl, t.Key(), t.Type, t.Payload, t.LastFailedAt, t.ErrorMsg) | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload), formatPastTime(t.LastFailedAt), t.LastErr) | ||||
| 			} | ||||
| 		}) | ||||
| } | ||||
|  | ||||
| func listCompletedTasks(qname string, pageNum, pageSize int) { | ||||
| 	i := createInspector() | ||||
| 	tasks, err := i.ListCompletedTasks(qname, asynq.PageSize(pageSize), asynq.Page(pageNum)) | ||||
| 	if err != nil { | ||||
| 		fmt.Println(err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	if len(tasks) == 0 { | ||||
| 		fmt.Printf("No completed tasks in %q queue\n", qname) | ||||
| 		return | ||||
| 	} | ||||
| 	printTable( | ||||
| 		[]string{"ID", "Type", "Payload", "CompletedAt", "Result"}, | ||||
| 		func(w io.Writer, tmpl string) { | ||||
| 			for _, t := range tasks { | ||||
| 				fmt.Fprintf(w, tmpl, t.ID, t.Type, sprintBytes(t.Payload), formatPastTime(t.CompletedAt), sprintBytes(t.Result)) | ||||
| 			} | ||||
| 		}) | ||||
| } | ||||
|  | ||||
| func taskCancel(cmd *cobra.Command, args []string) { | ||||
| 	r := createRDB() | ||||
| 	i := createInspector() | ||||
| 	for _, id := range args { | ||||
| 		err := r.PublishCancelation(id) | ||||
| 		if err != nil { | ||||
| 		if err := i.CancelProcessing(id); err != nil { | ||||
| 			fmt.Printf("error: could not send cancelation signal: %v\n", err) | ||||
| 			continue | ||||
| 		} | ||||
| @@ -305,25 +345,82 @@ func taskCancel(cmd *cobra.Command, args []string) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func taskKill(cmd *cobra.Command, args []string) { | ||||
| func taskInspect(cmd *cobra.Command, args []string) { | ||||
| 	qname, err := cmd.Flags().GetString("queue") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	key, err := cmd.Flags().GetString("key") | ||||
| 	id, err := cmd.Flags().GetString("id") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	i := createInspector() | ||||
| 	err = i.ArchiveTaskByKey(qname, key) | ||||
| 	info, err := i.GetTaskInfo(qname, id) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	fmt.Println("task transitioned to archived state") | ||||
| 	printTaskInfo(info) | ||||
| } | ||||
|  | ||||
| func printTaskInfo(info *asynq.TaskInfo) { | ||||
| 	bold := color.New(color.Bold) | ||||
| 	bold.Println("Task Info") | ||||
| 	fmt.Printf("Queue:   %s\n", info.Queue) | ||||
| 	fmt.Printf("ID:      %s\n", info.ID) | ||||
| 	fmt.Printf("Type:    %s\n", info.Type) | ||||
| 	fmt.Printf("State:   %v\n", info.State) | ||||
| 	fmt.Printf("Retried: %d/%d\n", info.Retried, info.MaxRetry) | ||||
| 	fmt.Println() | ||||
| 	fmt.Printf("Next process time: %s\n", formatNextProcessAt(info.NextProcessAt)) | ||||
| 	if len(info.LastErr) != 0 { | ||||
| 		fmt.Println() | ||||
| 		bold.Println("Last Failure") | ||||
| 		fmt.Printf("Failed at:     %s\n", formatPastTime(info.LastFailedAt)) | ||||
| 		fmt.Printf("Error message: %s\n", info.LastErr) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func formatNextProcessAt(processAt time.Time) string { | ||||
| 	if processAt.IsZero() || processAt.Unix() == 0 { | ||||
| 		return "n/a" | ||||
| 	} | ||||
| 	if processAt.Before(time.Now()) { | ||||
| 		return "now" | ||||
| 	} | ||||
| 	return fmt.Sprintf("%s (in %v)", processAt.Format(time.UnixDate), processAt.Sub(time.Now()).Round(time.Second)) | ||||
| } | ||||
|  | ||||
| // formatPastTime takes t which is time in the past and returns a user-friendly string. | ||||
| func formatPastTime(t time.Time) string { | ||||
| 	if t.IsZero() || t.Unix() == 0 { | ||||
| 		return "" | ||||
| 	} | ||||
| 	return t.Format(time.UnixDate) | ||||
| } | ||||
|  | ||||
| func taskArchive(cmd *cobra.Command, args []string) { | ||||
| 	qname, err := cmd.Flags().GetString("queue") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	id, err := cmd.Flags().GetString("id") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	i := createInspector() | ||||
| 	err = i.ArchiveTask(qname, id) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	fmt.Println("task archived") | ||||
| } | ||||
|  | ||||
| func taskDelete(cmd *cobra.Command, args []string) { | ||||
| @@ -332,14 +429,14 @@ func taskDelete(cmd *cobra.Command, args []string) { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	key, err := cmd.Flags().GetString("key") | ||||
| 	id, err := cmd.Flags().GetString("id") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	i := createInspector() | ||||
| 	err = i.DeleteTaskByKey(qname, key) | ||||
| 	err = i.DeleteTask(qname, id) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| @@ -353,19 +450,19 @@ func taskRun(cmd *cobra.Command, args []string) { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	key, err := cmd.Flags().GetString("key") | ||||
| 	id, err := cmd.Flags().GetString("id") | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
|  | ||||
| 	i := createInspector() | ||||
| 	err = i.RunTaskByKey(qname, key) | ||||
| 	err = i.RunTask(qname, id) | ||||
| 	if err != nil { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	fmt.Println("task transitioned to pending state") | ||||
| 	fmt.Println("task is now pending") | ||||
| } | ||||
|  | ||||
| func taskArchiveAll(cmd *cobra.Command, args []string) { | ||||
| @@ -383,6 +480,8 @@ func taskArchiveAll(cmd *cobra.Command, args []string) { | ||||
| 	i := createInspector() | ||||
| 	var n int | ||||
| 	switch state { | ||||
| 	case "pending": | ||||
| 		n, err = i.ArchiveAllPendingTasks(qname) | ||||
| 	case "scheduled": | ||||
| 		n, err = i.ArchiveAllScheduledTasks(qname) | ||||
| 	case "retry": | ||||
| @@ -395,7 +494,7 @@ func taskArchiveAll(cmd *cobra.Command, args []string) { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	fmt.Printf("%d tasks transitioned to archived state\n", n) | ||||
| 	fmt.Printf("%d tasks archived\n", n) | ||||
| } | ||||
|  | ||||
| func taskDeleteAll(cmd *cobra.Command, args []string) { | ||||
| @@ -413,12 +512,16 @@ func taskDeleteAll(cmd *cobra.Command, args []string) { | ||||
| 	i := createInspector() | ||||
| 	var n int | ||||
| 	switch state { | ||||
| 	case "pending": | ||||
| 		n, err = i.DeleteAllPendingTasks(qname) | ||||
| 	case "scheduled": | ||||
| 		n, err = i.DeleteAllScheduledTasks(qname) | ||||
| 	case "retry": | ||||
| 		n, err = i.DeleteAllRetryTasks(qname) | ||||
| 	case "archived": | ||||
| 		n, err = i.DeleteAllArchivedTasks(qname) | ||||
| 	case "completed": | ||||
| 		n, err = i.DeleteAllCompletedTasks(qname) | ||||
| 	default: | ||||
| 		fmt.Printf("error: unsupported state %q\n", state) | ||||
| 		os.Exit(1) | ||||
| @@ -459,5 +562,5 @@ func taskRunAll(cmd *cobra.Command, args []string) { | ||||
| 		fmt.Printf("error: %v\n", err) | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| 	fmt.Printf("%d tasks transitioned to pending state\n", n) | ||||
| 	fmt.Printf("%d tasks are now pending\n", n) | ||||
| } | ||||
|   | ||||
							
								
								
									
										14
									
								
								tools/go.mod
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								tools/go.mod
									
									
									
									
									
								
							| @@ -3,17 +3,13 @@ module github.com/hibiken/asynq/tools | ||||
| go 1.13 | ||||
|  | ||||
| require ( | ||||
| 	github.com/coreos/go-etcd v2.0.0+incompatible // indirect | ||||
| 	github.com/cpuguy83/go-md2man v1.0.10 // indirect | ||||
| 	github.com/fatih/color v1.9.0 | ||||
| 	github.com/go-redis/redis/v7 v7.4.0 | ||||
| 	github.com/google/uuid v1.1.1 | ||||
| 	github.com/hibiken/asynq v0.14.0 | ||||
| 	github.com/go-redis/redis/v8 v8.11.2 | ||||
| 	github.com/google/uuid v1.2.0 | ||||
| 	github.com/hibiken/asynq v0.17.1 | ||||
| 	github.com/mitchellh/go-homedir v1.1.0 | ||||
| 	github.com/spf13/cast v1.3.1 | ||||
| 	github.com/spf13/cobra v1.0.0 | ||||
| 	github.com/spf13/viper v1.6.2 | ||||
| 	github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8 // indirect | ||||
| 	github.com/spf13/cobra v1.1.1 | ||||
| 	github.com/spf13/viper v1.7.0 | ||||
| ) | ||||
|  | ||||
| replace github.com/hibiken/asynq => ./.. | ||||
|   | ||||
							
								
								
									
										293
									
								
								tools/go.sum
									
									
									
									
									
								
							
							
						
						
									
										293
									
								
								tools/go.sum
									
									
									
									
									
								
							| @@ -1,67 +1,132 @@ | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= | ||||
| cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= | ||||
| cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= | ||||
| cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= | ||||
| cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= | ||||
| cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= | ||||
| cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= | ||||
| cloud.google.com/go/firestore v1.1.0/go.mod h1:ulACoGHTpvq5r8rxGJ4ddJZBZqakUQqClKRT5SZwBmk= | ||||
| cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= | ||||
| cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw= | ||||
| dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | ||||
| github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | ||||
| github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= | ||||
| github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= | ||||
| github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= | ||||
| github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= | ||||
| github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= | ||||
| 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/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/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||
| github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk= | ||||
| github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
| github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= | ||||
| github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | ||||
| github.com/coreos/etcd v3.3.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
| github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= | ||||
| github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= | ||||
| github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= | ||||
| github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= | ||||
| github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= | ||||
| 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/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= | ||||
| 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/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV9I= | ||||
| 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/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= | ||||
| github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| 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-redis/redis/v7 v7.2.0 h1:CrCexy/jYWZjW0AyVoHlcJUeZN19VWlbepTh1Vq6dJs= | ||||
| github.com/go-redis/redis/v7 v7.2.0/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRfnwdHj/Dcg= | ||||
| github.com/go-redis/redis/v7 v7.4.0 h1:7obg6wUoj05T0EpY0o8B59S9w5yeMWql7sw2kwNW1x4= | ||||
| github.com/go-redis/redis/v7 v7.4.0/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRfnwdHj/Dcg= | ||||
| github.com/go-redis/redis/v8 v8.11.2 h1:WqlSpAwz8mxDSMCvbyz1Mkiqe0LE5OY4j3lgkvu1Ts0= | ||||
| github.com/go-redis/redis/v8 v8.11.2/go.mod h1:DLomh7y2e3ggQXQLd1YgmvIfecPJoFl7WU5SOQ/r06M= | ||||
| github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= | ||||
| github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= | ||||
| github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= | ||||
| github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||
| github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= | ||||
| github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||
| github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= | ||||
| 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 h1:6nsPYzhq5kReh6QImI3k5qWzO4PEbvbIW2cwSfR/6xs= | ||||
| 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 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| 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= | ||||
| github.com/google/go-cmp v0.4.0 h1:xsAVV57WRhGj6kEIi8ReJzQlHHqcBYCElAvkovg3B/4= | ||||
| 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/uuid v1.1.1 h1:Gkbcsh/GbpXz7lPftLA3P6TYMwjCLYm83jiFQZF/3gY= | ||||
| github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/go-cmp v0.5.0/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/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 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | ||||
| github.com/google/uuid v1.2.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= | ||||
| github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= | ||||
| github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= | ||||
| github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= | ||||
| github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= | ||||
| github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q= | ||||
| github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= | ||||
| github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||
| github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= | ||||
| github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= | ||||
| github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= | ||||
| github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= | ||||
| github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= | ||||
| github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= | ||||
| github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= | ||||
| github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90= | ||||
| github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= | ||||
| github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= | ||||
| github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= | ||||
| github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI= | ||||
| github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= | ||||
| github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ= | ||||
| 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/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= | ||||
| github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= | ||||
| github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= | ||||
| github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||
| github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= | ||||
| github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= | ||||
| github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | ||||
| github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= | ||||
| @@ -74,35 +139,54 @@ github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORN | ||||
| 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/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | ||||
| 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-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/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= | ||||
| github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= | ||||
| github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= | ||||
| github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= | ||||
| github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||
| github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= | ||||
| github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= | ||||
| github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= | ||||
| github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE= | ||||
| github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| 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/nxadm/tail v1.4.4 h1:DQuhQpB1tVlglWS2hLQ5OV6B5r8aGxSrPc5Qo6uTN78= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| 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.10.1 h1:q/mM8GF/n0shIN8SaAZ0V+jnLPzen6WIVZdiwrRlMlo= | ||||
| github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/gomega v1.7.0 h1:XPnZz8VVBHjVsy1vzJmRwIcSwiUO+JFfrv/xGiigmME= | ||||
| github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.15.0 h1:1V1NfVQR87RtWAgp1lv9JZJ5Jap+XFGKPi00andXGi4= | ||||
| github.com/onsi/ginkgo v1.15.0/go.mod h1:hF8qUzuuC8DJGygJH3726JnCZX4MYbRB8yFfISqnKUg= | ||||
| 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 h1:7n6FEkpFmfCoo2t+YYqXH0evK+a9ICQz0xcAy9dYcaQ= | ||||
| github.com/onsi/gomega v1.10.5/go.mod h1:gza4q3jKQJijlu05nKWRCW/GavJumGt8aNRxWg7mt48= | ||||
| github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= | ||||
| github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc= | ||||
| github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= | ||||
| 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/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= | ||||
| github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= | ||||
| github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso= | ||||
| 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/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro= | ||||
| github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| @@ -111,8 +195,10 @@ github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40T | ||||
| 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/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= | ||||
| github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= | ||||
| 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= | ||||
| github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= | ||||
| github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= | ||||
| github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d h1:zE9ykElWQ6/NYmHa3jpm/yHnI4xSofP+UP6SpjHcSeM= | ||||
| @@ -126,84 +212,184 @@ github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B | ||||
| 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/cobra v0.0.5 h1:f0B+LkLX6DtmRH1isoNA9VTtNUK9K8xYd28JNNfOv/s= | ||||
| github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= | ||||
| github.com/spf13/cobra v1.0.0 h1:6m/oheQuQ13N9ks4hubMG6BnvwOeaJrqSPLahSnczz8= | ||||
| github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE= | ||||
| 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= | ||||
| github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= | ||||
| github.com/spf13/pflag v1.0.3 h1:zPAT6CGy6wXeQ7NtTnaTerfKOsV6V6F8agHXFiazDkg= | ||||
| github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= | ||||
| github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= | ||||
| github.com/spf13/viper v1.4.0/go.mod h1:PTJ7Z/lr49W6bUbkmS1V3by4uWynFiR9p7+dSq/yZzE= | ||||
| github.com/spf13/viper v1.6.2 h1:7aKfF+e8/k68gda3LOjo5RxiUqddoFxVq4BKBPrxk5E= | ||||
| github.com/spf13/viper v1.6.2/go.mod h1:t3iDnF5Jlj76alVNuyFBk5oUMCvsrkbvZK0WQdfDi5k= | ||||
| github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= | ||||
| github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||
| github.com/spf13/viper v1.7.0 h1:xVKxvI7ouOI5I+U9s2eeiUfMaWBVoXA3AWskkrqK0VM= | ||||
| github.com/spf13/viper v1.7.0/go.mod h1:8WkrPz2fc9jxqZNCJI/76HCieCp4Q8HaLFoCha5qpdg= | ||||
| 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 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= | ||||
| 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.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/subosito/gotenv v1.2.0 h1:Slr1R9HxAlEKefgq5jn9U+DnETlIUa6HfgEzj0g5d7s= | ||||
| github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc= | ||||
| github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= | ||||
| github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= | ||||
| github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| 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/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= | ||||
| golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/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-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | ||||
| golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= | ||||
| golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= | ||||
| golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= | ||||
| golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= | ||||
| golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| 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-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||
| golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | ||||
| golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= | ||||
| golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= | ||||
| golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= | ||||
| golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= | ||||
| 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-20181023162649-9b4f9f5ad519/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-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20181220203305-927f97764cc3/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-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= | ||||
| golang.org/x/net v0.0.0-20190923162816-aa69164e4478 h1:l5EDrHhldLYb3ZRHDUhXF7Om7MvYXnkV9/iQNo1lX6g= | ||||
| golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= | ||||
| 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-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb h1:eBmm0M9fYhWpKZLjQUUKka/LtIxf46G4fxeEz5KJr9U= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
| golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= | ||||
| golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/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-20190227155943-e225da77a7e6/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-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| 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-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| 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-20181205085412-a5c9d58dba9a/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-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| 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-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| 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-20191204072324-ce4227a45e2e h1:9vRrk9YW2BTzLP0VCB9ZDjU4cPqkg+IDWL7XgxA1yxQ= | ||||
| golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091 h1:DMyOG0U+gKfu8JZzg2UQe9MeaC1X+xQWlAKcRnjxjCw= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs= | ||||
| golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/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 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| 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/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= | ||||
| 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-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= | ||||
| golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | ||||
| golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= | ||||
| golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/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= | ||||
| google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= | ||||
| google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= | ||||
| google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||
| google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | ||||
| google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= | ||||
| google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||
| google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= | ||||
| google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | ||||
| google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= | ||||
| google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= | ||||
| 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.21.0/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= | ||||
| google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= | ||||
| google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= | ||||
| 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 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c= | ||||
| google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/fsnotify.v1 v1.4.7 h1:xOHLXZwVvI9hhs+cLKq5+I5onOuwQLhQwiu63xxlHs4= | ||||
| gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/ini.v1 v1.51.0 h1:AQvPpx3LzTDM0AjnIRlVFwFFGC+npRopjZxLJj6gdno= | ||||
| gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| @@ -212,8 +398,15 @@ gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkep | ||||
| 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= | ||||
| 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.7 h1:VUgggvou5XRW9mHwD/yXxIYSMtY0zoKQf/v226p2nyo= | ||||
| gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| 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-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||
| honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= | ||||
| rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= | ||||
|   | ||||
							
								
								
									
										40
									
								
								x/rate/example_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								x/rate/example_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| package rate_test | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	"github.com/hibiken/asynq/x/rate" | ||||
| ) | ||||
|  | ||||
| type RateLimitError struct { | ||||
| 	RetryIn time.Duration | ||||
| } | ||||
|  | ||||
| func (e *RateLimitError) Error() string { | ||||
| 	return fmt.Sprintf("rate limited (retry in  %v)", e.RetryIn) | ||||
| } | ||||
|  | ||||
| func ExampleNewSemaphore() { | ||||
| 	redisConnOpt := asynq.RedisClientOpt{Addr: ":6379"} | ||||
| 	sema := rate.NewSemaphore(redisConnOpt, "my_queue", 10) | ||||
| 	// call sema.Close() when appropriate | ||||
|  | ||||
| 	_ = asynq.HandlerFunc(func(ctx context.Context, task *asynq.Task) error { | ||||
| 		ok, err := sema.Acquire(ctx) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if !ok { | ||||
| 			return &RateLimitError{RetryIn: 30 * time.Second} | ||||
| 		} | ||||
|  | ||||
| 		// Make sure to release the token once we're done. | ||||
| 		defer sema.Release(ctx) | ||||
|  | ||||
| 		// Process task | ||||
| 		return nil | ||||
| 	}) | ||||
| } | ||||
							
								
								
									
										114
									
								
								x/rate/semaphore.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								x/rate/semaphore.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | ||||
| // Package rate contains rate limiting strategies for asynq.Handler(s). | ||||
| package rate | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/go-redis/redis/v8" | ||||
| 	"github.com/hibiken/asynq" | ||||
| 	asynqcontext "github.com/hibiken/asynq/internal/context" | ||||
| ) | ||||
|  | ||||
| // NewSemaphore creates a counting Semaphore for the given scope with the given number of tokens. | ||||
| func NewSemaphore(rco asynq.RedisConnOpt, scope string, maxTokens int) *Semaphore { | ||||
| 	rc, ok := rco.MakeRedisClient().(redis.UniversalClient) | ||||
| 	if !ok { | ||||
| 		panic(fmt.Sprintf("rate.NewSemaphore: unsupported RedisConnOpt type %T", rco)) | ||||
| 	} | ||||
|  | ||||
| 	if maxTokens < 1 { | ||||
| 		panic("rate.NewSemaphore: maxTokens cannot be less than 1") | ||||
| 	} | ||||
|  | ||||
| 	if len(strings.TrimSpace(scope)) == 0 { | ||||
| 		panic("rate.NewSemaphore: scope should not be empty") | ||||
| 	} | ||||
|  | ||||
| 	return &Semaphore{ | ||||
| 		rc:        rc, | ||||
| 		scope:     scope, | ||||
| 		maxTokens: maxTokens, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Semaphore is a distributed counting semaphore which can be used to set maxTokens across multiple asynq servers. | ||||
| type Semaphore struct { | ||||
| 	rc        redis.UniversalClient | ||||
| 	maxTokens int | ||||
| 	scope     string | ||||
| } | ||||
|  | ||||
| // KEYS[1] -> asynq:sema:<scope> | ||||
| // ARGV[1] -> max concurrency | ||||
| // ARGV[2] -> current time in unix time | ||||
| // ARGV[3] -> deadline in unix time | ||||
| // ARGV[4] -> task ID | ||||
| var acquireCmd = redis.NewScript(` | ||||
| redis.call("ZREMRANGEBYSCORE", KEYS[1], "-inf", tonumber(ARGV[2])-1) | ||||
| local count = redis.call("ZCARD", KEYS[1]) | ||||
|  | ||||
| if (count < tonumber(ARGV[1])) then | ||||
|      redis.call("ZADD", KEYS[1], ARGV[3], ARGV[4]) | ||||
|      return 'true' | ||||
| else | ||||
|      return 'false' | ||||
| end | ||||
| `) | ||||
|  | ||||
| // Acquire attempts to acquire a token from the semaphore. | ||||
| // - Returns (true, nil), iff semaphore key exists and current value is less than maxTokens | ||||
| // - Returns (false, nil) when token cannot be acquired | ||||
| // - Returns (false, error) otherwise | ||||
| // | ||||
| // The context.Context passed to Acquire must have a deadline set, | ||||
| // this ensures that token is released if the job goroutine crashes and does not call Release. | ||||
| func (s *Semaphore) Acquire(ctx context.Context) (bool, error) { | ||||
| 	d, ok := ctx.Deadline() | ||||
| 	if !ok { | ||||
| 		return false, fmt.Errorf("provided context must have a deadline") | ||||
| 	} | ||||
|  | ||||
| 	taskID, ok := asynqcontext.GetTaskID(ctx) | ||||
| 	if !ok { | ||||
| 		return false, fmt.Errorf("provided context is missing task ID value") | ||||
| 	} | ||||
|  | ||||
| 	return acquireCmd.Run(ctx, s.rc, | ||||
| 		[]string{semaphoreKey(s.scope)}, | ||||
| 		s.maxTokens, | ||||
| 		time.Now().Unix(), | ||||
| 		d.Unix(), | ||||
| 		taskID, | ||||
| 	).Bool() | ||||
| } | ||||
|  | ||||
| // Release will release the token on the counting semaphore. | ||||
| func (s *Semaphore) Release(ctx context.Context) error { | ||||
| 	taskID, ok := asynqcontext.GetTaskID(ctx) | ||||
| 	if !ok { | ||||
| 		return fmt.Errorf("provided context is missing task ID value") | ||||
| 	} | ||||
|  | ||||
| 	n, err := s.rc.ZRem(ctx, semaphoreKey(s.scope), taskID).Result() | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("redis command failed: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if n == 0 { | ||||
| 		return fmt.Errorf("no token found for task %q", taskID) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Close closes the connection to redis. | ||||
| func (s *Semaphore) Close() error { | ||||
| 	return s.rc.Close() | ||||
| } | ||||
|  | ||||
| func semaphoreKey(scope string) string { | ||||
| 	return fmt.Sprintf("asynq:sema:%s", scope) | ||||
| } | ||||
							
								
								
									
										408
									
								
								x/rate/semaphore_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										408
									
								
								x/rate/semaphore_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,408 @@ | ||||
| package rate | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"flag" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
| 	"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" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	redisAddr string | ||||
| 	redisDB   int | ||||
|  | ||||
| 	useRedisCluster   bool | ||||
| 	redisClusterAddrs string // comma-separated list of host:port | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	flag.StringVar(&redisAddr, "redis_addr", "localhost:6379", "redis address to use in testing") | ||||
| 	flag.IntVar(&redisDB, "redis_db", 14, "redis db number to use in testing") | ||||
| 	flag.BoolVar(&useRedisCluster, "redis_cluster", false, "use redis cluster as a broker in testing") | ||||
| 	flag.StringVar(&redisClusterAddrs, "redis_cluster_addrs", "localhost:7000,localhost:7001,localhost:7002", "comma separated list of redis server addresses") | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc           string | ||||
| 		name           string | ||||
| 		maxConcurrency int | ||||
| 		wantPanic      string | ||||
| 		connOpt        asynq.RedisConnOpt | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:      "Bad RedisConnOpt", | ||||
| 			wantPanic: "rate.NewSemaphore: unsupported RedisConnOpt type *rate.badConnOpt", | ||||
| 			connOpt:   &badConnOpt{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:      "Zero maxTokens should panic", | ||||
| 			wantPanic: "rate.NewSemaphore: maxTokens cannot be less than 1", | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:           "Empty scope should panic", | ||||
| 			maxConcurrency: 2, | ||||
| 			name:           "    ", | ||||
| 			wantPanic:      "rate.NewSemaphore: scope should not be empty", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.desc, func(t *testing.T) { | ||||
| 			if tt.wantPanic != "" { | ||||
| 				defer func() { | ||||
| 					if r := recover(); r.(string) != tt.wantPanic { | ||||
| 						t.Errorf("%s;\nNewSemaphore should panic with msg: %s, got %s", tt.desc, tt.wantPanic, r.(string)) | ||||
| 					} | ||||
| 				}() | ||||
| 			} | ||||
|  | ||||
| 			opt := tt.connOpt | ||||
| 			if tt.connOpt == nil { | ||||
| 				opt = getRedisConnOpt(t) | ||||
| 			} | ||||
|  | ||||
| 			sema := NewSemaphore(opt, tt.name, tt.maxConcurrency) | ||||
| 			defer sema.Close() | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore_Acquire(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc           string | ||||
| 		name           string | ||||
| 		maxConcurrency int | ||||
| 		taskIDs        []string | ||||
| 		ctxFunc        func(string) (context.Context, context.CancelFunc) | ||||
| 		want           []bool | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:           "Should acquire token when current token count is less than maxTokens", | ||||
| 			name:           "task-1", | ||||
| 			maxConcurrency: 3, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-1", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| 			}, | ||||
| 			want: []bool{true, true}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:           "Should fail acquiring token when current token count is equal to maxTokens", | ||||
| 			name:           "task-2", | ||||
| 			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{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-2", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| 			}, | ||||
| 			want: []bool{true, true, true, false}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.desc, func(t *testing.T) { | ||||
| 			opt := getRedisConnOpt(t) | ||||
| 			rc := opt.MakeRedisClient().(redis.UniversalClient) | ||||
| 			defer rc.Close() | ||||
|  | ||||
| 			if err := rc.Del(context.Background(), semaphoreKey(tt.name)).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			sema := NewSemaphore(opt, tt.name, tt.maxConcurrency) | ||||
| 			defer sema.Close() | ||||
|  | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				ctx, cancel := tt.ctxFunc(tt.taskIDs[i]) | ||||
|  | ||||
| 				got, err := sema.Acquire(ctx) | ||||
| 				if err != nil { | ||||
| 					t.Errorf("%s;\nSemaphore.Acquire() got error %v", tt.desc, err) | ||||
| 				} | ||||
|  | ||||
| 				if got != tt.want[i] { | ||||
| 					t.Errorf("%s;\nSemaphore.Acquire(ctx) returned %v, want %v", tt.desc, got, tt.want[i]) | ||||
| 				} | ||||
|  | ||||
| 				cancel() | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore_Acquire_Error(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc           string | ||||
| 		name           string | ||||
| 		maxConcurrency int | ||||
| 		taskIDs        []string | ||||
| 		ctxFunc        func(string) (context.Context, context.CancelFunc) | ||||
| 		errStr         string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:           "Should return error if context has no deadline", | ||||
| 			name:           "task-3", | ||||
| 			maxConcurrency: 1, | ||||
| 			taskIDs:        []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return context.Background(), func() {} | ||||
| 			}, | ||||
| 			errStr: "provided context must have a deadline", | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:           "Should return error when context is missing taskID", | ||||
| 			name:           "task-4", | ||||
| 			maxConcurrency: 1, | ||||
| 			taskIDs:        []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||
| 				return context.WithTimeout(context.Background(), time.Second) | ||||
| 			}, | ||||
| 			errStr: "provided context is missing task ID value", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.desc, func(t *testing.T) { | ||||
| 			opt := getRedisConnOpt(t) | ||||
| 			rc := opt.MakeRedisClient().(redis.UniversalClient) | ||||
| 			defer rc.Close() | ||||
|  | ||||
| 			if err := rc.Del(context.Background(), semaphoreKey(tt.name)).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			sema := NewSemaphore(opt, tt.name, tt.maxConcurrency) | ||||
| 			defer sema.Close() | ||||
|  | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				ctx, cancel := tt.ctxFunc(tt.taskIDs[i]) | ||||
|  | ||||
| 				_, err := sema.Acquire(ctx) | ||||
| 				if err == nil || err.Error() != tt.errStr { | ||||
| 					t.Errorf("%s;\nSemaphore.Acquire() got error %v want error %v", tt.desc, err, tt.errStr) | ||||
| 				} | ||||
|  | ||||
| 				cancel() | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore_Acquire_StaleToken(t *testing.T) { | ||||
| 	opt := getRedisConnOpt(t) | ||||
| 	rc := opt.MakeRedisClient().(redis.UniversalClient) | ||||
| 	defer rc.Close() | ||||
|  | ||||
| 	taskID := uuid.NewString() | ||||
|  | ||||
| 	// 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{ | ||||
| 		Score:  float64(time.Now().Add(-10 * time.Second).Unix()), | ||||
| 		Member: taskID, | ||||
| 	}) | ||||
|  | ||||
| 	sema := NewSemaphore(opt, "stale-token", 1) | ||||
| 	defer sema.Close() | ||||
|  | ||||
| 	ctx, cancel := asynqcontext.New(&base.TaskMessage{ | ||||
| 		ID:    taskID, | ||||
| 		Queue: "task-1", | ||||
| 	}, time.Now().Add(time.Second)) | ||||
| 	defer cancel() | ||||
|  | ||||
| 	got, err := sema.Acquire(ctx) | ||||
| 	if err != nil { | ||||
| 		t.Errorf("Acquire_StaleToken;\nSemaphore.Acquire() got error %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if !got { | ||||
| 		t.Error("Acquire_StaleToken;\nSemaphore.Acquire() got false want true") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore_Release(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		desc      string | ||||
| 		name      string | ||||
| 		taskIDs   []string | ||||
| 		ctxFunc   func(string) (context.Context, context.CancelFunc) | ||||
| 		wantCount int64 | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:    "Should decrease token count", | ||||
| 			name:    "task-5", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-3", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should decrease token count by 2", | ||||
| 			name:    "task-6", | ||||
| 			taskIDs: []string{uuid.NewString(), uuid.NewString()}, | ||||
| 			ctxFunc: func(id string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    id, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.desc, func(t *testing.T) { | ||||
| 			opt := getRedisConnOpt(t) | ||||
| 			rc := opt.MakeRedisClient().(redis.UniversalClient) | ||||
| 			defer rc.Close() | ||||
|  | ||||
| 			if err := rc.Del(context.Background(), semaphoreKey(tt.name)).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []*redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, &redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
| 			} | ||||
| 			if err := rc.ZAdd(context.Background(), semaphoreKey(tt.name), members...).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.ZAdd() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			sema := NewSemaphore(opt, tt.name, 3) | ||||
| 			defer sema.Close() | ||||
|  | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				ctx, cancel := tt.ctxFunc(tt.taskIDs[i]) | ||||
|  | ||||
| 				if err := sema.Release(ctx); err != nil { | ||||
| 					t.Errorf("%s;\nSemaphore.Release() got error %v", tt.desc, err) | ||||
| 				} | ||||
|  | ||||
| 				cancel() | ||||
| 			} | ||||
|  | ||||
| 			i, err := rc.ZCount(context.Background(), semaphoreKey(tt.name), "-inf", "+inf").Result() | ||||
| 			if err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.ZCount() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			if i != tt.wantCount { | ||||
| 				t.Errorf("%s;\nSemaphore.Release(ctx) didn't release token, got %v want 0", tt.desc, i) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewSemaphore_Release_Error(t *testing.T) { | ||||
| 	testID := uuid.NewString() | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		desc    string | ||||
| 		name    string | ||||
| 		taskIDs []string | ||||
| 		ctxFunc func(string) (context.Context, context.CancelFunc) | ||||
| 		errStr  string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			desc:    "Should return error when context is missing taskID", | ||||
| 			name:    "task-7", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||
| 				return context.WithTimeout(context.Background(), time.Second) | ||||
| 			}, | ||||
| 			errStr: "provided context is missing task ID value", | ||||
| 		}, | ||||
| 		{ | ||||
| 			desc:    "Should return error when context has taskID which never acquired token", | ||||
| 			name:    "task-8", | ||||
| 			taskIDs: []string{uuid.NewString()}, | ||||
| 			ctxFunc: func(_ string) (context.Context, context.CancelFunc) { | ||||
| 				return asynqcontext.New(&base.TaskMessage{ | ||||
| 					ID:    testID, | ||||
| 					Queue: "task-4", | ||||
| 				}, time.Now().Add(time.Second)) | ||||
| 			}, | ||||
| 			errStr: fmt.Sprintf("no token found for task %q", testID), | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.desc, func(t *testing.T) { | ||||
| 			opt := getRedisConnOpt(t) | ||||
| 			rc := opt.MakeRedisClient().(redis.UniversalClient) | ||||
| 			defer rc.Close() | ||||
|  | ||||
| 			if err := rc.Del(context.Background(), semaphoreKey(tt.name)).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.Del() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			var members []*redis.Z | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				members = append(members, &redis.Z{ | ||||
| 					Score:  float64(time.Now().Add(time.Duration(i) * time.Second).Unix()), | ||||
| 					Member: tt.taskIDs[i], | ||||
| 				}) | ||||
| 			} | ||||
| 			if err := rc.ZAdd(context.Background(), semaphoreKey(tt.name), members...).Err(); err != nil { | ||||
| 				t.Errorf("%s;\nredis.UniversalClient.ZAdd() got error %v", tt.desc, err) | ||||
| 			} | ||||
|  | ||||
| 			sema := NewSemaphore(opt, tt.name, 3) | ||||
| 			defer sema.Close() | ||||
|  | ||||
| 			for i := 0; i < len(tt.taskIDs); i++ { | ||||
| 				ctx, cancel := tt.ctxFunc(tt.taskIDs[i]) | ||||
|  | ||||
| 				if err := sema.Release(ctx); err == nil || err.Error() != tt.errStr { | ||||
| 					t.Errorf("%s;\nSemaphore.Release() got error %v want error %v", tt.desc, err, tt.errStr) | ||||
| 				} | ||||
|  | ||||
| 				cancel() | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func getRedisConnOpt(tb testing.TB) asynq.RedisConnOpt { | ||||
| 	tb.Helper() | ||||
| 	if useRedisCluster { | ||||
| 		addrs := strings.Split(redisClusterAddrs, ",") | ||||
| 		if len(addrs) == 0 { | ||||
| 			tb.Fatal("No redis cluster addresses provided. Please set addresses using --redis_cluster_addrs flag.") | ||||
| 		} | ||||
| 		return asynq.RedisClusterClientOpt{ | ||||
| 			Addrs: addrs, | ||||
| 		} | ||||
| 	} | ||||
| 	return asynq.RedisClientOpt{ | ||||
| 		Addr: redisAddr, | ||||
| 		DB:   redisDB, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type badConnOpt struct { | ||||
| } | ||||
|  | ||||
| func (b badConnOpt) MakeRedisClient() interface{} { | ||||
| 	return nil | ||||
| } | ||||
		Reference in New Issue
	
	Block a user