mirror of
				https://github.com/aykhans/dodo.git
				synced 2025-10-25 01:40:58 +00:00 
			
		
		
		
	Compare commits
	
		
			30 Commits
		
	
	
		
			v0.7.0
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cabe562d47 | ||
| 25d4762a3c | |||
| 361d423651 | |||
| ffa724fae7 | |||
| 7930be490d | |||
| e6c54e9cb2 | |||
| b32f567de7 | |||
| b6e85d9443 | |||
| 827e3535cd | |||
| 7ecf534d87 | |||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 17ad5fadb9 | ||
| 7fb59a7989 | |||
| 527909c882 | |||
| 4459675efa | |||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 604af355e6 | ||
| 7d4267c4c2 | |||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 845ab7296c | ||
| 49d004ff06 | |||
| 045deb6120 | |||
| 075ef26203 | |||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 946afbb2c3 | ||
| aacb33cfa5 | |||
| 4a7db48351 | |||
| b73087dce5 | |||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 20a46feab8 | ||
| 0adde6e04e | |||
| ca50de4e2f | |||
| c99e7c66d9 | |||
| 280e5f5c4e | |||
| 47dfad6046 | 
							
								
								
									
										2
									
								
								.github/workflows/golangci-lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/golangci-lint.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,5 +21,5 @@ jobs: | ||||
|       - name: golangci-lint | ||||
|         uses: golangci/golangci-lint-action@v7 | ||||
|         with: | ||||
|           version: v2.0.2 | ||||
|           version: v2.4.0 | ||||
|           args: --timeout=10m --config=.golangci.yml | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| version: "2" | ||||
|  | ||||
| run: | ||||
|     go: "1.24" | ||||
|     go: "1.25" | ||||
|     concurrency: 8 | ||||
|     timeout: 10m | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| FROM golang:1.24-alpine AS builder | ||||
| FROM golang:1.25-alpine AS builder | ||||
|  | ||||
| WORKDIR /src | ||||
|  | ||||
|   | ||||
							
								
								
									
										934
									
								
								EXAMPLES.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										934
									
								
								EXAMPLES.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,934 @@ | ||||
| # Dodo Usage Examples | ||||
|  | ||||
| This document provides comprehensive examples of using Dodo with various configuration combinations. Each example includes three methods: CLI usage, YAML configuration, and JSON configuration. | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| 1. [Basic HTTP Stress Testing](#1-basic-http-stress-testing) | ||||
| 2. [POST Request with Form Data](#2-post-request-with-form-data) | ||||
| 3. [API Testing with Authentication](#3-api-testing-with-authentication) | ||||
| 4. [Testing with Custom Headers and Cookies](#4-testing-with-custom-headers-and-cookies) | ||||
| 5. [Load Testing with Proxy Rotation](#5-load-testing-with-proxy-rotation) | ||||
| 6. [JSON API Testing with Dynamic Data](#6-json-api-testing-with-dynamic-data) | ||||
| 7. [File Upload Testing](#7-file-upload-testing) | ||||
| 8. [E-commerce Cart Testing](#8-e-commerce-cart-testing) | ||||
| 9. [GraphQL API Testing](#9-graphql-api-testing) | ||||
| 10. [WebSocket-style HTTP Testing](#10-websocket-style-http-testing) | ||||
| 11. [Multi-tenant Application Testing](#11-multi-tenant-application-testing) | ||||
| 12. [Rate Limiting Testing](#12-rate-limiting-testing) | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 1. Basic HTTP Stress Testing | ||||
|  | ||||
| Test a simple website with basic GET requests to measure performance under load. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/get \ | ||||
|      -m GET \ | ||||
|      -d 5 \ | ||||
|      -r 100 \ | ||||
|      -t 5s \ | ||||
|      -o 30s \ | ||||
|      --skip-verify=false \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://httpbin.org/get" | ||||
| yes: true | ||||
| timeout: "5s" | ||||
| dodos: 5 | ||||
| requests: 100 | ||||
| duration: "30s" | ||||
| skip_verify: false | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://httpbin.org/get", | ||||
|     "yes": true, | ||||
|     "timeout": "5s", | ||||
|     "dodos": 5, | ||||
|     "requests": 100, | ||||
|     "duration": "30s", | ||||
|     "skip_verify": false | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 2. POST Request with Form Data | ||||
|  | ||||
| Test form submission endpoints with randomized form data. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/post \ | ||||
|      -m POST \ | ||||
|      -d 3 \ | ||||
|      -r 50 \ | ||||
|      -t 10s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Content-Type:application/x-www-form-urlencoded" \ | ||||
|      -b "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 12 }}&email={{ fakeit_Email }}" \ | ||||
|      -b "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 8 }}&email={{ fakeit_Email }}" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://httpbin.org/post" | ||||
| yes: true | ||||
| timeout: "10s" | ||||
| dodos: 3 | ||||
| requests: 50 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Content-Type: "application/x-www-form-urlencoded" | ||||
|  | ||||
| body: | ||||
|     - "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 12 }}&email={{ fakeit_Email }}" | ||||
|     - "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 8 }}&email={{ fakeit_Email }}" | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://httpbin.org/post", | ||||
|     "yes": true, | ||||
|     "timeout": "10s", | ||||
|     "dodos": 3, | ||||
|     "requests": 50, | ||||
|     "skip_verify": false, | ||||
|     "headers": [{ "Content-Type": "application/x-www-form-urlencoded" }], | ||||
|     "body": [ | ||||
|         "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 12 }}&email={{ fakeit_Email }}", | ||||
|         "username={{ fakeit_Username }}&password={{ fakeit_Password true true true true true 8 }}&email={{ fakeit_Email }}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 3. API Testing with Authentication | ||||
|  | ||||
| Test protected API endpoints with various authentication methods. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/bearer \ | ||||
|      -m GET \ | ||||
|      -d 4 \ | ||||
|      -r 200 \ | ||||
|      -t 8s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Authorization:Bearer {{ fakeit_LetterN 32 }}" \ | ||||
|      -H "User-Agent:{{ fakeit_UserAgent }}" \ | ||||
|      -H "X-Request-ID:{{ fakeit_Int }}" \ | ||||
|      -H "Accept:application/json" \ | ||||
|      -p "api_version=v1" \ | ||||
|      -p "format=json" \ | ||||
|      -p "client_id=mobile" -p "client_id=web" -p "client_id=desktop" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://httpbin.org/bearer" | ||||
| yes: true | ||||
| timeout: "8s" | ||||
| dodos: 4 | ||||
| requests: 200 | ||||
| skip_verify: false | ||||
|  | ||||
| params: | ||||
|     - api_version: "v1" | ||||
|     - format: "json" | ||||
|     - client_id: ["mobile", "web", "desktop"] | ||||
|  | ||||
| headers: | ||||
|     - Authorization: "Bearer {{ fakeit_LetterN 32 }}" | ||||
|     - User-Agent: "{{ fakeit_UserAgent }}" | ||||
|     - X-Request-ID: "{{ fakeit_Int }}" | ||||
|     - Accept: "application/json" | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://httpbin.org/bearer", | ||||
|     "yes": true, | ||||
|     "timeout": "8s", | ||||
|     "dodos": 4, | ||||
|     "requests": 200, | ||||
|     "skip_verify": false, | ||||
|     "params": [ | ||||
|         { "api_version": "v1" }, | ||||
|         { "format": "json" }, | ||||
|         { "client_id": ["mobile", "web", "desktop"] } | ||||
|     ], | ||||
|     "headers": [ | ||||
|         { "Authorization": "Bearer {{ fakeit_LetterN 32 }}" }, | ||||
|         { "User-Agent": "{{ fakeit_UserAgent }}" }, | ||||
|         { "X-Request-ID": "{{ fakeit_Int }}" }, | ||||
|         { "Accept": "application/json" } | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 4. Testing with Custom Headers and Cookies | ||||
|  | ||||
| Test applications that require specific headers and session cookies. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/cookies \ | ||||
|      -m GET \ | ||||
|      -d 6 \ | ||||
|      -r 75 \ | ||||
|      -t 5s \ | ||||
|      --skip-verify=false \ | ||||
|      -H 'Accept-Language:{{ strings_Join "," (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) }}' \ | ||||
|      -H "X-Forwarded-For:{{ fakeit_IPv4Address }}" \ | ||||
|      -H "X-Real-IP:{{ fakeit_IPv4Address }}" \ | ||||
|      -H "Accept-Encoding:gzip" -H "Accept-Encoding:deflate" -H "Accept-Encoding:br" \ | ||||
|      -c "session_id={{ fakeit_UUID }}" \ | ||||
|      -c 'user_pref={{ fakeit_RandomString "a1" "b2" "c3" }}' \ | ||||
|      -c "theme=dark" -c "theme=light" -c "theme=auto" \ | ||||
|      -c "lang=en" -c "lang=es" -c "lang=fr" -c "lang=de" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://httpbin.org/cookies" | ||||
| yes: true | ||||
| timeout: "5s" | ||||
| dodos: 6 | ||||
| requests: 75 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Accept-Language: '{{ strings_Join "," (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) }}' | ||||
|     - X-Forwarded-For: "{{ fakeit_IPv4Address }}" | ||||
|     - X-Real-IP: "{{ fakeit_IPv4Address }}" | ||||
|     - Accept-Encoding: ["gzip", "deflate", "br"] | ||||
|  | ||||
| cookies: | ||||
|     - session_id: "{{ fakeit_UUID }}" | ||||
|     - user_pref: '{{ fakeit_RandomString "a1" "b2" "c3" }}' | ||||
|     - theme: ["dark", "light", "auto"] | ||||
|     - lang: ["en", "es", "fr", "de"] | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://httpbin.org/cookies", | ||||
|     "yes": true, | ||||
|     "timeout": "5s", | ||||
|     "dodos": 6, | ||||
|     "requests": 75, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { | ||||
|             "Accept-Language": "{{ strings_Join \",\" (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) (fakeit_LanguageAbbreviation) }}" | ||||
|         }, | ||||
|         { "X-Forwarded-For": "{{ fakeit_IPv4Address }}" }, | ||||
|         { "X-Real-IP": "{{ fakeit_IPv4Address }}" }, | ||||
|         { "Accept-Encoding": ["gzip", "deflate", "br"] } | ||||
|     ], | ||||
|     "cookies": [ | ||||
|         { "session_id": "{{ fakeit_UUID }}" }, | ||||
|         { "user_pref": "{{ fakeit_RandomString \"a1\" \"b2\" \"c3\" }}" }, | ||||
|         { "theme": ["dark", "light", "auto"] }, | ||||
|         { "lang": ["en", "es", "fr", "de"] } | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 5. Load Testing with Proxy Rotation | ||||
|  | ||||
| Test through multiple proxies for distributed load testing. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/ip \ | ||||
|      -m GET \ | ||||
|      -d 8 \ | ||||
|      -r 300 \ | ||||
|      -t 15s \ | ||||
|      --skip-verify=false \ | ||||
|      -x "http://proxy1.example.com:8080" \ | ||||
|      -x "http://proxy2.example.com:8080" \ | ||||
|      -x "socks5://proxy3.example.com:1080" \ | ||||
|      -x "http://username:password@proxy4.example.com:8080" \ | ||||
|      -H "User-Agent:{{ fakeit_UserAgent }}" \ | ||||
|      -H "Accept:application/json" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://httpbin.org/ip" | ||||
| yes: true | ||||
| timeout: "15s" | ||||
| dodos: 8 | ||||
| requests: 300 | ||||
| skip_verify: false | ||||
|  | ||||
| proxy: | ||||
|     - "http://proxy1.example.com:8080" | ||||
|     - "http://proxy2.example.com:8080" | ||||
|     - "socks5://proxy3.example.com:1080" | ||||
|     - "http://username:password@proxy4.example.com:8080" | ||||
|  | ||||
| headers: | ||||
|     - User-Agent: "{{ fakeit_UserAgent }}" | ||||
|     - Accept: "application/json" | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://httpbin.org/ip", | ||||
|     "yes": true, | ||||
|     "timeout": "15s", | ||||
|     "dodos": 8, | ||||
|     "requests": 300, | ||||
|     "skip_verify": false, | ||||
|     "proxy": [ | ||||
|         "http://proxy1.example.com:8080", | ||||
|         "http://proxy2.example.com:8080", | ||||
|         "socks5://proxy3.example.com:1080", | ||||
|         "http://username:password@proxy4.example.com:8080" | ||||
|     ], | ||||
|     "headers": [ | ||||
|         { "User-Agent": "{{ fakeit_UserAgent }}" }, | ||||
|         { "Accept": "application/json" } | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 6. JSON API Testing with Dynamic Data | ||||
|  | ||||
| Test REST APIs with realistic JSON payloads. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/post \ | ||||
|      -m POST \ | ||||
|      -d 5 \ | ||||
|      -r 150 \ | ||||
|      -t 12s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Content-Type:application/json" \ | ||||
|      -H "Accept:application/json" \ | ||||
|      -H "X-API-Version:2023-10-01" \ | ||||
|      -b '{"user_id":{{ fakeit_Uint }},"name":"{{ fakeit_Name }}","email":"{{ fakeit_Email }}","created_at":"{{ fakeit_Date }}"}' \ | ||||
|      -b '{"product_id":{{ fakeit_Uint }},"name":"{{ fakeit_ProductName }}","price":{{ fakeit_Price 10 1000 }},"category":"{{ fakeit_ProductCategory }}"}' \ | ||||
|      -b '{"order_id":"{{ fakeit_UUID }}","items":[{"id":{{ fakeit_Uint }},"quantity":{{ fakeit_IntRange 1 10 }}}],"total":{{ fakeit_Price 50 500 }}}' \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://httpbin.org/post" | ||||
| yes: true | ||||
| timeout: "12s" | ||||
| dodos: 5 | ||||
| requests: 150 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Content-Type: "application/json" | ||||
|     - Accept: "application/json" | ||||
|     - X-API-Version: "2023-10-01" | ||||
|  | ||||
| body: | ||||
|     - '{"user_id":{{ fakeit_Uint }},"name":"{{ fakeit_Name }}","email":"{{ fakeit_Email }}","created_at":"{{ fakeit_Date }}"}' | ||||
|     - '{"product_id":{{ fakeit_Uint }},"name":"{{ fakeit_ProductName }}","price":{{ fakeit_Price 10 1000 }},"category":"{{ fakeit_ProductCategory }}"}' | ||||
|     - '{"order_id":"{{ fakeit_UUID }}","items":[{"id":{{ fakeit_Uint }},"quantity":{{ fakeit_IntRange 1 10 }}}],"total":{{ fakeit_Price 50 500 }}}' | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://httpbin.org/post", | ||||
|     "yes": true, | ||||
|     "timeout": "12s", | ||||
|     "dodos": 5, | ||||
|     "requests": 150, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "Content-Type": "application/json" }, | ||||
|         { "Accept": "application/json" }, | ||||
|         { "X-API-Version": "2023-10-01" } | ||||
|     ], | ||||
|     "body": [ | ||||
|         "{\"user_id\":{{ fakeit_Uint }},\"name\":\"{{ fakeit_Name }}\",\"email\":\"{{ fakeit_Email }}\",\"created_at\":\"{{ fakeit_Date }}\"}", | ||||
|         "{\"product_id\":{{ fakeit_Uint }},\"name\":\"{{ fakeit_ProductName }}\",\"price\":{{ fakeit_Price 10 1000 }},\"category\":\"{{ fakeit_ProductCategory }}\"}", | ||||
|         "{\"order_id\":\"{{ fakeit_UUID }}\",\"items\":[{\"id\":{{ fakeit_Uint }},\"quantity\":{{ fakeit_IntRange 1 10 }}}],\"total\":{{ fakeit_Price 50 500 }}}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 7. File Upload Testing | ||||
|  | ||||
| Test file upload endpoints with multipart form data. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://httpbin.org/post \ | ||||
|      -m POST \ | ||||
|      -d 3 \ | ||||
|      -r 25 \ | ||||
|      -t 30s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "X-Upload-Source:dodo-test" \ | ||||
|      -H "User-Agent:{{ fakeit_UserAgent }}" \ | ||||
|      -b '{{ body_FormData (dict_Str "filename" (fakeit_UUID) "content" (fakeit_Paragraph 3 5 10 " ")) }}' \ | ||||
|      -b '{{ body_FormData (dict_Str "file" (fakeit_UUID) "description" (fakeit_Sentence 10) "category" "image") }}' \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://httpbin.org/post" | ||||
| yes: true | ||||
| timeout: "30s" | ||||
| dodos: 3 | ||||
| requests: 25 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - X-Upload-Source: "dodo-test" | ||||
|     - User-Agent: "{{ fakeit_UserAgent }}" | ||||
|  | ||||
| body: | ||||
|     - '{{ body_FormData (dict_Str "filename" (fakeit_UUID) "content" (fakeit_Paragraph 3 5 10 " ")) }}' | ||||
|     - '{{ body_FormData (dict_Str "file" (fakeit_UUID) "description" (fakeit_Sentence 10) "category" "image") }}' | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://httpbin.org/post", | ||||
|     "yes": true, | ||||
|     "timeout": "30s", | ||||
|     "dodos": 3, | ||||
|     "requests": 25, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "X-Upload-Source": "dodo-test" }, | ||||
|         { "User-Agent": "{{ fakeit_UserAgent }}" } | ||||
|     ], | ||||
|     "body": [ | ||||
|         "{{ body_FormData (dict_Str \"filename\" (fakeit_UUID) \"content\" (fakeit_Paragraph 3 5 10 \" \")) }}", | ||||
|         "{{ body_FormData (dict_Str \"file\" (fakeit_UUID) \"description\" (fakeit_Sentence 10) \"category\" \"image\") }}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 8. E-commerce Cart Testing | ||||
|  | ||||
| Test shopping cart operations with realistic product data. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://api.example-shop.com/cart \ | ||||
|      -m POST \ | ||||
|      -d 10 \ | ||||
|      -r 500 \ | ||||
|      -t 8s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Content-Type:application/json" \ | ||||
|      -H "Authorization:Bearer {{ fakeit_LetterN 32 }}" \ | ||||
|      -H "X-Client-Version:1.2.3" \ | ||||
|      -H "User-Agent:{{ fakeit_UserAgent }}" \ | ||||
|      -c "cart_session={{ fakeit_UUID }}" \ | ||||
|      -c "user_pref=guest" -c "user_pref=member" -c "user_pref=premium" \ | ||||
|      -c "region=US" -c "region=EU" -c "region=ASIA" \ | ||||
|      -p "currency=USD" -p "currency=EUR" -p "currency=GBP" \ | ||||
|      -p "locale=en-US" -p "locale=en-GB" -p "locale=de-DE" -p "locale=fr-FR" \ | ||||
|      -b '{"action":"add","product_id":"{{ fakeit_UUID }}","quantity":{{ fakeit_IntRange 1 5 }},"user_id":"{{ fakeit_UUID }}"}' \ | ||||
|      -b '{"action":"remove","product_id":"{{ fakeit_UUID }}","user_id":"{{ fakeit_UUID }}"}' \ | ||||
|      -b '{"action":"update","product_id":"{{ fakeit_UUID }}","quantity":{{ fakeit_IntRange 1 10 }},"user_id":"{{ fakeit_UUID }}"}' \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://api.example-shop.com/cart" | ||||
| yes: true | ||||
| timeout: "8s" | ||||
| dodos: 10 | ||||
| requests: 500 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Content-Type: "application/json" | ||||
|     - Authorization: "Bearer {{ fakeit_LetterN 32 }}" | ||||
|     - X-Client-Version: "1.2.3" | ||||
|     - User-Agent: "{{ fakeit_UserAgent }}" | ||||
|  | ||||
| cookies: | ||||
|     - cart_session: "{{ fakeit_UUID }}" | ||||
|     - user_pref: ["guest", "member", "premium"] | ||||
|     - region: ["US", "EU", "ASIA"] | ||||
|  | ||||
| params: | ||||
|     - currency: ["USD", "EUR", "GBP"] | ||||
|     - locale: ["en-US", "en-GB", "de-DE", "fr-FR"] | ||||
|  | ||||
| body: | ||||
|     - '{"action":"add","product_id":"{{ fakeit_UUID }}","quantity":{{ fakeit_IntRange 1 5 }},"user_id":"{{ fakeit_UUID }}"}' | ||||
|     - '{"action":"remove","product_id":"{{ fakeit_UUID }}","user_id":"{{ fakeit_UUID }}"}' | ||||
|     - '{"action":"update","product_id":"{{ fakeit_UUID }}","quantity":{{ fakeit_IntRange 1 10 }},"user_id":"{{ fakeit_UUID }}"}' | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://api.example-shop.com/cart", | ||||
|     "yes": true, | ||||
|     "timeout": "8s", | ||||
|     "dodos": 10, | ||||
|     "requests": 500, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "Content-Type": "application/json" }, | ||||
|         { "Authorization": "Bearer {{ fakeit_LetterN 32 }}" }, | ||||
|         { "X-Client-Version": "1.2.3" }, | ||||
|         { "User-Agent": "{{ fakeit_UserAgent }}" } | ||||
|     ], | ||||
|     "cookies": [ | ||||
|         { "cart_session": "{{ fakeit_UUID }}" }, | ||||
|         { "user_pref": ["guest", "member", "premium"] }, | ||||
|         { "region": ["US", "EU", "ASIA"] } | ||||
|     ], | ||||
|     "params": [ | ||||
|         { "currency": ["USD", "EUR", "GBP"] }, | ||||
|         { "locale": ["en-US", "en-GB", "de-DE", "fr-FR"] } | ||||
|     ], | ||||
|     "body": [ | ||||
|         "{\"action\":\"add\",\"product_id\":\"{{ fakeit_UUID }}\",\"quantity\":{{ fakeit_IntRange 1 5 }},\"user_id\":\"{{ fakeit_UUID }}\"}", | ||||
|         "{\"action\":\"remove\",\"product_id\":\"{{ fakeit_UUID }}\",\"user_id\":\"{{ fakeit_UUID }}\"}", | ||||
|         "{\"action\":\"update\",\"product_id\":\"{{ fakeit_UUID }}\",\"quantity\":{{ fakeit_IntRange 1 10 }},\"user_id\":\"{{ fakeit_UUID }}\"}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 9. GraphQL API Testing | ||||
|  | ||||
| Test GraphQL endpoints with various queries and mutations. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://api.example.com/graphql \ | ||||
|      -m POST \ | ||||
|      -d 4 \ | ||||
|      -r 100 \ | ||||
|      -t 10s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Content-Type:application/json" \ | ||||
|      -H "Authorization:Bearer {{ fakeit_UUID }}" \ | ||||
|      -H "X-GraphQL-Client:dodo-test" \ | ||||
|      -b '{"query":"query GetUser($id: ID!) { user(id: $id) { id name email } }","variables":{"id":"{{ fakeit_UUID }}"}}' \ | ||||
|      -b '{"query":"query GetPosts($limit: Int) { posts(limit: $limit) { id title content } }","variables":{"limit":{{ fakeit_IntRange 5 20 }}}}' \ | ||||
|      -b '{"query":"mutation CreatePost($input: PostInput!) { createPost(input: $input) { id title } }","variables":{"input":{"title":"{{ fakeit_Sentence 5 }}","content":"{{ fakeit_Paragraph 2 3 5 " "}}","authorId":"{{ fakeit_UUID }}"}}}' \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://api.example.com/graphql" | ||||
| yes: true | ||||
| timeout: "10s" | ||||
| dodos: 4 | ||||
| requests: 100 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Content-Type: "application/json" | ||||
|     - Authorization: "Bearer {{ fakeit_UUID }}" | ||||
|     - X-GraphQL-Client: "dodo-test" | ||||
|  | ||||
| body: | ||||
|     - '{"query":"query GetUser($id: ID!) { user(id: $id) { id name email } }","variables":{"id":"{{ fakeit_UUID }}"}}' | ||||
|     - '{"query":"query GetPosts($limit: Int) { posts(limit: $limit) { id title content } }","variables":{"limit":{{ fakeit_IntRange 5 20 }}}}' | ||||
|     - '{"query":"mutation CreatePost($input: PostInput!) { createPost(input: $input) { id title } }","variables":{"input":{"title":"{{ fakeit_Sentence 5 }}","content":"{{ fakeit_Paragraph 2 3 5 " "}}","authorId":"{{ fakeit_UUID }}"}}}' | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://api.example.com/graphql", | ||||
|     "yes": true, | ||||
|     "timeout": "10s", | ||||
|     "dodos": 4, | ||||
|     "requests": 100, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "Content-Type": "application/json" }, | ||||
|         { "Authorization": "Bearer {{ fakeit_UUID }}" }, | ||||
|         { "X-GraphQL-Client": "dodo-test" } | ||||
|     ], | ||||
|     "body": [ | ||||
|         "{\"query\":\"query GetUser($id: ID!) { user(id: $id) { id name email } }\",\"variables\":{\"id\":\"{{ fakeit_UUID }}\"}}", | ||||
|         "{\"query\":\"query GetPosts($limit: Int) { posts(limit: $limit) { id title content } }\",\"variables\":{\"limit\":{{ fakeit_IntRange 5 20 }}}}", | ||||
|         "{\"query\":\"mutation CreatePost($input: PostInput!) { createPost(input: $input) { id title } }\",\"variables\":{\"input\":{\"title\":\"{{ fakeit_Sentence 5 }}\",\"content\":\"{{ fakeit_Paragraph 2 3 5 \\\" \\\"}}\",\"authorId\":\"{{ fakeit_UUID }}\"}}}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 10. WebSocket-style HTTP Testing | ||||
|  | ||||
| Test real-time applications with WebSocket-like HTTP endpoints. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://api.realtime-app.com/events \ | ||||
|      -m POST \ | ||||
|      -d 15 \ | ||||
|      -r 1000 \ | ||||
|      -t 5s \ | ||||
|      -o 60s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "Content-Type:application/json" \ | ||||
|      -H "X-Event-Type:{{ fakeit_LetterNN 4 12 }}" \ | ||||
|      -H "Connection:keep-alive" \ | ||||
|      -H "Cache-Control:no-cache" \ | ||||
|      -c "connection_id={{ fakeit_UUID }}" \ | ||||
|      -c "session_token={{ fakeit_UUID }}" \ | ||||
|      -p "channel=general" -p "channel=notifications" -p "channel=alerts" -p "channel=updates" \ | ||||
|      -p "version=v1" -p "version=v2" \ | ||||
|      -b '{"event":"{{ fakeit_Word }}","data":{"timestamp":"{{ fakeit_Date }}","user_id":"{{ fakeit_UUID }}","message":"{{ fakeit_Sentence 8 }}"}}' \ | ||||
|      -b '{"event":"ping","data":{"timestamp":"{{ fakeit_Date }}","client_id":"{{ fakeit_UUID }}"}}' \ | ||||
|      -b '{"event":"status_update","data":{"status":"{{ fakeit_Word }}","user_id":"{{ fakeit_UUID }}","timestamp":"{{ fakeit_Date }}"}}' \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "POST" | ||||
| url: "https://api.realtime-app.com/events" | ||||
| yes: true | ||||
| timeout: "5s" | ||||
| dodos: 15 | ||||
| requests: 1000 | ||||
| duration: "60s" | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - Content-Type: "application/json" | ||||
|     - X-Event-Type: "{{ fakeit_LetterNN 4 12 }}" | ||||
|     - Connection: "keep-alive" | ||||
|     - Cache-Control: "no-cache" | ||||
|  | ||||
| cookies: | ||||
|     - connection_id: "{{ fakeit_UUID }}" | ||||
|     - session_token: "{{ fakeit_UUID }}" | ||||
|  | ||||
| params: | ||||
|     - channel: ["general", "notifications", "alerts", "updates"] | ||||
|     - version: ["v1", "v2"] | ||||
|  | ||||
| body: | ||||
|     - '{"event":"{{ fakeit_Word }}","data":{"timestamp":"{{ fakeit_Date }}","user_id":"{{ fakeit_UUID }}","message":"{{ fakeit_Sentence 8 }}"}}' | ||||
|     - '{"event":"ping","data":{"timestamp":"{{ fakeit_Date }}","client_id":"{{ fakeit_UUID }}"}}' | ||||
|     - '{"event":"status_update","data":{"status":"{{ fakeit_Word }}","user_id":"{{ fakeit_UUID }}","timestamp":"{{ fakeit_Date }}"}}' | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "POST", | ||||
|     "url": "https://api.realtime-app.com/events", | ||||
|     "yes": true, | ||||
|     "timeout": "5s", | ||||
|     "dodos": 15, | ||||
|     "requests": 1000, | ||||
|     "duration": "60s", | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "Content-Type": "application/json" }, | ||||
|         { "X-Event-Type": "{{ fakeit_LetterNN 4 12 }}" }, | ||||
|         { "Connection": "keep-alive" }, | ||||
|         { "Cache-Control": "no-cache" } | ||||
|     ], | ||||
|     "cookies": [ | ||||
|         { "connection_id": "{{ fakeit_UUID }}" }, | ||||
|         { "session_token": "{{ fakeit_UUID }}" } | ||||
|     ], | ||||
|     "params": [ | ||||
|         { "channel": ["general", "notifications", "alerts", "updates"] }, | ||||
|         { "version": ["v1", "v2"] } | ||||
|     ], | ||||
|     "body": [ | ||||
|         "{\"event\":\"{{ fakeit_Word }}\",\"data\":{\"timestamp\":\"{{ fakeit_Date }}\",\"user_id\":\"{{ fakeit_UUID }}\",\"message\":\"{{ fakeit_Sentence 8 }}\"}}", | ||||
|         "{\"event\":\"ping\",\"data\":{\"timestamp\":\"{{ fakeit_Date }}\",\"client_id\":\"{{ fakeit_UUID }}\"}}", | ||||
|         "{\"event\":\"status_update\",\"data\":{\"status\":\"{{ fakeit_Word }}\",\"user_id\":\"{{ fakeit_UUID }}\",\"timestamp\":\"{{ fakeit_Date }}\"}}" | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 11. Multi-tenant Application Testing | ||||
|  | ||||
| Test SaaS applications with tenant-specific configurations. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://app.saas-platform.com/api/data \ | ||||
|      -m GET \ | ||||
|      -d 12 \ | ||||
|      -r 600 \ | ||||
|      -t 15s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "X-Tenant-ID:{{ fakeit_UUID }}" \ | ||||
|      -H "Authorization:Bearer {{ fakeit_LetterN 64 }}" \ | ||||
|      -H "X-Client-Type:web" -H "X-Client-Type:mobile" -H "X-Client-Type:api" \ | ||||
|      -H "Accept:application/json" \ | ||||
|      -c "tenant_session={{ fakeit_UUID }}" \ | ||||
|      -c "user_role=admin" -c "user_role=user" -c "user_role=viewer" \ | ||||
|      -c "subscription_tier=free" -c "subscription_tier=pro" -c "subscription_tier=enterprise" \ | ||||
|      -p "page={{ fakeit_IntRange 1 10 }}" \ | ||||
|      -p "limit={{ fakeit_IntRange 10 100 }}" \ | ||||
|      -p "sort=created_at" -p "sort=updated_at" -p "sort=name" \ | ||||
|      -p "order=asc" -p "order=desc" \ | ||||
|      -p "filter_by=active" -p "filter_by=inactive" -p "filter_by=pending" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://app.saas-platform.com/api/data" | ||||
| yes: true | ||||
| timeout: "15s" | ||||
| dodos: 12 | ||||
| requests: 600 | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - X-Tenant-ID: "{{ fakeit_UUID }}" | ||||
|     - Authorization: "Bearer {{ fakeit_LetterN 64 }}" | ||||
|     - X-Client-Type: ["web", "mobile", "api"] | ||||
|     - Accept: "application/json" | ||||
|  | ||||
| cookies: | ||||
|     - tenant_session: "{{ fakeit_UUID }}" | ||||
|     - user_role: ["admin", "user", "viewer"] | ||||
|     - subscription_tier: ["free", "pro", "enterprise"] | ||||
|  | ||||
| params: | ||||
|     - page: "{{ fakeit_IntRange 1 10 }}" | ||||
|     - limit: "{{ fakeit_IntRange 10 100 }}" | ||||
|     - sort: ["created_at", "updated_at", "name"] | ||||
|     - order: ["asc", "desc"] | ||||
|     - filter_by: ["active", "inactive", "pending"] | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://app.saas-platform.com/api/data", | ||||
|     "yes": true, | ||||
|     "timeout": "15s", | ||||
|     "dodos": 12, | ||||
|     "requests": 600, | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "X-Tenant-ID": "{{ fakeit_UUID }}" }, | ||||
|         { "Authorization": "Bearer {{ fakeit_LetterN 64 }}" }, | ||||
|         { "X-Client-Type": ["web", "mobile", "api"] }, | ||||
|         { "Accept": "application/json" } | ||||
|     ], | ||||
|     "cookies": [ | ||||
|         { "tenant_session": "{{ fakeit_UUID }}" }, | ||||
|         { "user_role": ["admin", "user", "viewer"] }, | ||||
|         { "subscription_tier": ["free", "pro", "enterprise"] } | ||||
|     ], | ||||
|     "params": [ | ||||
|         { "page": "{{ fakeit_IntRange 1 10 }}" }, | ||||
|         { "limit": "{{ fakeit_IntRange 10 100 }}" }, | ||||
|         { "sort": ["created_at", "updated_at", "name"] }, | ||||
|         { "order": ["asc", "desc"] }, | ||||
|         { "filter_by": ["active", "inactive", "pending"] } | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 12. Rate Limiting Testing | ||||
|  | ||||
| Test API rate limits and throttling mechanisms. | ||||
|  | ||||
| ### CLI Usage | ||||
|  | ||||
| ```bash | ||||
| dodo -u https://api.rate-limited.com/endpoint \ | ||||
|      -m GET \ | ||||
|      -d 20 \ | ||||
|      -r 2000 \ | ||||
|      -t 3s \ | ||||
|      -o 120s \ | ||||
|      --skip-verify=false \ | ||||
|      -H "X-API-Key:{{ fakeit_UUID }}" \ | ||||
|      -H "X-Client-ID:{{ fakeit_UUID }}" \ | ||||
|      -H "X-Rate-Limit-Test:true" \ | ||||
|      -H "User-Agent:{{ fakeit_UserAgent }}" \ | ||||
|      -c "rate_limit_bucket={{ fakeit_UUID }}" \ | ||||
|      -c "client_tier=tier1" -c "client_tier=tier2" -c "client_tier=tier3" \ | ||||
|      -p "burst_test=true" \ | ||||
|      -p "client_type=premium" -p "client_type=standard" -p "client_type=free" \ | ||||
|      -p "request_id={{ fakeit_UUID }}" \ | ||||
|      -y | ||||
| ``` | ||||
|  | ||||
| ### YAML Configuration | ||||
|  | ||||
| ```yaml | ||||
| method: "GET" | ||||
| url: "https://api.rate-limited.com/endpoint" | ||||
| yes: true | ||||
| timeout: "3s" | ||||
| dodos: 20 | ||||
| requests: 2000 | ||||
| duration: "120s" | ||||
| skip_verify: false | ||||
|  | ||||
| headers: | ||||
|     - X-API-Key: "{{ fakeit_UUID }}" | ||||
|     - X-Client-ID: "{{ fakeit_UUID }}" | ||||
|     - X-Rate-Limit-Test: "true" | ||||
|     - User-Agent: "{{ fakeit_UserAgent }}" | ||||
|  | ||||
| params: | ||||
|     - burst_test: "true" | ||||
|     - client_type: ["premium", "standard", "free"] | ||||
|     - request_id: "{{ fakeit_UUID }}" | ||||
|  | ||||
| cookies: | ||||
|     - rate_limit_bucket: "{{ fakeit_UUID }}" | ||||
|     - client_tier: ["tier1", "tier2", "tier3"] | ||||
| ``` | ||||
|  | ||||
| ### JSON Configuration | ||||
|  | ||||
| ```json | ||||
| { | ||||
|     "method": "GET", | ||||
|     "url": "https://api.rate-limited.com/endpoint", | ||||
|     "yes": true, | ||||
|     "timeout": "3s", | ||||
|     "dodos": 20, | ||||
|     "requests": 2000, | ||||
|     "duration": "120s", | ||||
|     "skip_verify": false, | ||||
|     "headers": [ | ||||
|         { "X-API-Key": "{{ fakeit_UUID }}" }, | ||||
|         { "X-Client-ID": "{{ fakeit_UUID }}" }, | ||||
|         { "X-Rate-Limit-Test": "true" }, | ||||
|         { "User-Agent": "{{ fakeit_UserAgent }}" } | ||||
|     ], | ||||
|     "params": [ | ||||
|         { "burst_test": "true" }, | ||||
|         { "client_type": ["premium", "standard", "free"] }, | ||||
|         { "request_id": "{{ fakeit_UUID }}" } | ||||
|     ], | ||||
|     "cookies": [ | ||||
|         { "rate_limit_bucket": "{{ fakeit_UUID }}" }, | ||||
|         { "client_tier": ["tier1", "tier2", "tier3"] } | ||||
|     ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Notes | ||||
|  | ||||
| - All examples use template functions for dynamic data generation | ||||
| - Adjust `dodos`, `requests`, `duration`, and `timeout` values based on your testing requirements | ||||
| - Use `skip_verify: true` for testing with self-signed certificates | ||||
| - Set `yes: true` to skip confirmation prompts in automated testing | ||||
| - Template functions like `{{ fakeit_* }}` generate random realistic data for each request | ||||
| - Multiple values in arrays (e.g., `["value1", "value2"]`) will be randomly selected per request | ||||
| - Use the `body_FormData` function for multipart form uploads | ||||
| - Proxy configurations support HTTP, SOCKS5, and SOCKS5H protocols | ||||
|  | ||||
| For more template functions and advanced configuration options, refer to the main documentation and `utils/templates.go`. | ||||
							
								
								
									
										27
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,7 +1,26 @@ | ||||
| <h1 align="center">Dodo - A Fast and Easy-to-Use HTTP Benchmarking Tool</h1> | ||||
| <p align="center"> | ||||
| <img width="30%" height="30%" src="https://ftp.aykhans.me/web/client/pubshares/VzPtSHS7yPQT7ngoZzZSNU/browse?path=%2Fdodo.png"> | ||||
| </p> | ||||
|  | ||||
|  | ||||
|  | ||||
| <div align="center"> | ||||
|   <h4> | ||||
|       <a href="./EXAMPLES.md"> | ||||
|           Examples | ||||
|       </a> | ||||
|       <span> | </span> | ||||
|       <a href="#installation"> | ||||
|           Install | ||||
|       </a> | ||||
|       <span> | </span> | ||||
|       <a href="https://hub.docker.com/r/aykhans/dodo"> | ||||
|           Docker | ||||
|       </a> | ||||
|   </h4> | ||||
|   <br> | ||||
|     <a href="https://coff.ee/aykhan"> | ||||
|         <img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 40px !important;width: 150px !important;"> | ||||
|     </a> | ||||
| </div> | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| @@ -245,6 +264,8 @@ With Docker: | ||||
| docker run --rm -i -v /path/to/config.json:/config.json aykhans/dodo -f /config.json -u https://example.com -m GET -d 10 -r 1000 -o 1m -t 5s | ||||
| ``` | ||||
|  | ||||
| You can find more usage examples in the [EXAMPLES.md](./EXAMPLES.md) file. | ||||
|  | ||||
| ## Config Parameters Reference | ||||
|  | ||||
| If `Headers`, `Params`, `Cookies`, `Body`, or `Proxy` fields have multiple values, each request will choose a random value from the list. | ||||
|   | ||||
| @@ -18,7 +18,7 @@ import ( | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	VERSION             string        = "0.7.0" | ||||
| 	VERSION             string        = "0.7.3" | ||||
| 	DefaultUserAgent    string        = "Dodo/" + VERSION | ||||
| 	DefaultMethod       string        = "GET" | ||||
| 	DefaultTimeout      time.Duration = time.Second * 10 | ||||
| @@ -159,9 +159,6 @@ func (config *Config) Validate() []error { | ||||
| 	if utils.IsNilOrZero(config.URL) { | ||||
| 		errs = append(errs, errors.New("request URL is required")) | ||||
| 	} else { | ||||
| 		if config.URL.Scheme == "" { | ||||
| 			config.URL.Scheme = "http" | ||||
| 		} | ||||
| 		if config.URL.Scheme != "http" && config.URL.Scheme != "https" { | ||||
| 			errs = append(errs, errors.New("request URL scheme must be http or https")) | ||||
| 		} | ||||
|   | ||||
							
								
								
									
										20
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,22 +1,22 @@ | ||||
| module github.com/aykhans/dodo | ||||
|  | ||||
| go 1.24.2 | ||||
| go 1.25 | ||||
|  | ||||
| require ( | ||||
| 	github.com/brianvoe/gofakeit/v7 v7.2.1 | ||||
| 	github.com/jedib0t/go-pretty/v6 v6.6.7 | ||||
| 	github.com/valyala/fasthttp v1.62.0 | ||||
| 	github.com/brianvoe/gofakeit/v7 v7.3.0 | ||||
| 	github.com/jedib0t/go-pretty/v6 v6.6.8 | ||||
| 	github.com/valyala/fasthttp v1.68.0 | ||||
| 	gopkg.in/yaml.v3 v3.0.1 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/andybalholm/brotli v1.1.1 // indirect | ||||
| 	github.com/klauspost/compress v1.18.0 // indirect | ||||
| 	github.com/andybalholm/brotli v1.2.0 // indirect | ||||
| 	github.com/klauspost/compress v1.18.1 // indirect | ||||
| 	github.com/mattn/go-runewidth v0.0.16 // indirect | ||||
| 	github.com/rivo/uniseg v0.4.7 // indirect | ||||
| 	github.com/valyala/bytebufferpool v1.0.0 // indirect | ||||
| 	golang.org/x/net v0.40.0 // indirect | ||||
| 	golang.org/x/sys v0.33.0 // indirect | ||||
| 	golang.org/x/term v0.32.0 // indirect | ||||
| 	golang.org/x/text v0.25.0 // indirect | ||||
| 	golang.org/x/net v0.46.0 // indirect | ||||
| 	golang.org/x/sys v0.37.0 // indirect | ||||
| 	golang.org/x/term v0.36.0 // indirect | ||||
| 	golang.org/x/text v0.30.0 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										36
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | ||||
| github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA= | ||||
| github.com/andybalholm/brotli v1.1.1/go.mod h1:05ib4cKhjx3OQYUY22hTVd34Bc8upXjOLL2rKwwZBoA= | ||||
| github.com/brianvoe/gofakeit/v7 v7.2.1 h1:AGojgaaCdgq4Adzrd2uWdbGNDyX6MWNhHdQBraNfOHI= | ||||
| github.com/brianvoe/gofakeit/v7 v7.2.1/go.mod h1:QXuPeBw164PJCzCUZVmgpgHJ3Llj49jSLVkKPMtxtxA= | ||||
| github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ= | ||||
| github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY= | ||||
| github.com/brianvoe/gofakeit/v7 v7.3.0 h1:TWStf7/lLpAjKw+bqwzeORo9jvrxToWEwp9b1J2vApQ= | ||||
| github.com/brianvoe/gofakeit/v7 v7.3.0/go.mod h1:QXuPeBw164PJCzCUZVmgpgHJ3Llj49jSLVkKPMtxtxA= | ||||
| 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/jedib0t/go-pretty/v6 v6.6.7 h1:m+LbHpm0aIAPLzLbMfn8dc3Ht8MW7lsSO4MPItz/Uuo= | ||||
| github.com/jedib0t/go-pretty/v6 v6.6.7/go.mod h1:YwC5CE4fJ1HFUDeivSV1r//AmANFHyqczZk+U6BDALU= | ||||
| github.com/klauspost/compress v1.18.0 h1:c/Cqfb0r+Yi+JtIEq73FWXVkRonBlf0CRNYc8Zttxdo= | ||||
| github.com/klauspost/compress v1.18.0/go.mod h1:2Pp+KzxcywXVXMr50+X0Q/Lsb43OQHYWRCY2AiWywWQ= | ||||
| github.com/jedib0t/go-pretty/v6 v6.6.8 h1:JnnzQeRz2bACBobIaa/r+nqjvws4yEhcmaZ4n1QzsEc= | ||||
| github.com/jedib0t/go-pretty/v6 v6.6.8/go.mod h1:YwC5CE4fJ1HFUDeivSV1r//AmANFHyqczZk+U6BDALU= | ||||
| github.com/klauspost/compress v1.18.1 h1:bcSGx7UbpBqMChDtsF28Lw6v/G94LPrrbMbdC3JH2co= | ||||
| github.com/klauspost/compress v1.18.1/go.mod h1:ZQFFVG+MdnR0P+l6wpXgIL4NTtwiKIdBnrBd8Nrxr+0= | ||||
| github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc= | ||||
| github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| @@ -19,18 +19,18 @@ github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOf | ||||
| github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= | ||||
| github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= | ||||
| github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= | ||||
| github.com/valyala/fasthttp v1.62.0 h1:8dKRBX/y2rCzyc6903Zu1+3qN0H/d2MsxPPmVNamiH0= | ||||
| github.com/valyala/fasthttp v1.62.0/go.mod h1:FCINgr4GKdKqV8Q0xv8b+UxPV+H/O5nNFo3D+r54Htg= | ||||
| github.com/valyala/fasthttp v1.68.0 h1:v12Nx16iepr8r9ySOwqI+5RBJ/DqTxhOy1HrHoDFnok= | ||||
| github.com/valyala/fasthttp v1.68.0/go.mod h1:5EXiRfYQAoiO/khu4oU9VISC/eVY6JqmSpPJoHCKsz4= | ||||
| github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU= | ||||
| github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E= | ||||
| golang.org/x/net v0.40.0 h1:79Xs7wF06Gbdcg4kdCCIQArK11Z1hr5POQ6+fIYHNuY= | ||||
| golang.org/x/net v0.40.0/go.mod h1:y0hY0exeL2Pku80/zKK7tpntoX23cqL3Oa6njdgRtds= | ||||
| golang.org/x/sys v0.33.0 h1:q3i8TbbEz+JRD9ywIRlyRAQbM0qF7hu24q3teo2hbuw= | ||||
| golang.org/x/sys v0.33.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= | ||||
| golang.org/x/term v0.32.0 h1:DR4lr0TjUs3epypdhTOkMmuF5CDFJ/8pOnbzMZPQ7bg= | ||||
| golang.org/x/term v0.32.0/go.mod h1:uZG1FhGx848Sqfsq4/DlJr3xGGsYMu/L5GW4abiaEPQ= | ||||
| golang.org/x/text v0.25.0 h1:qVyWApTSYLk/drJRO5mDlNYskwQznZmkpV2c8q9zls4= | ||||
| golang.org/x/text v0.25.0/go.mod h1:WEdwpYrmk1qmdHvhkSTNPm3app7v4rsT8F2UD6+VHIA= | ||||
| golang.org/x/net v0.46.0 h1:giFlY12I07fugqwPuWJi68oOnpfqFnJIJzaIIm2JVV4= | ||||
| golang.org/x/net v0.46.0/go.mod h1:Q9BGdFy1y4nkUwiLvT5qtyhAnEHgnQ/zd8PfU6nc210= | ||||
| golang.org/x/sys v0.37.0 h1:fdNQudmxPjkdUTPnLn5mdQv7Zwvbvpaxqs831goi9kQ= | ||||
| golang.org/x/sys v0.37.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= | ||||
| golang.org/x/term v0.36.0 h1:zMPR+aF8gfksFprF/Nc/rd1wRS1EI6nDBGyWAvDzx2Q= | ||||
| golang.org/x/term v0.36.0/go.mod h1:Qu394IJq6V6dCBRgwqshf3mPF85AqzYEzofzRdZkWss= | ||||
| golang.org/x/text v0.30.0 h1:yznKA/E9zq54KzlzBEAWn1NXSQ8DIp/NYMy88xJjl4k= | ||||
| golang.org/x/text v0.30.0/go.mod h1:yDdHFIX9t+tORqspjENWgzaCVXgk0yYnYuSZ8UzzBVM= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
|   | ||||
| @@ -14,47 +14,30 @@ type Response struct { | ||||
| 	Time     time.Duration | ||||
| } | ||||
|  | ||||
| type Responses []*Response | ||||
| type Responses []Response | ||||
|  | ||||
| // Print prints the responses in a tabular format, including information such as | ||||
| // response count, minimum time, maximum time, average time, and latency percentiles. | ||||
| func (responses Responses) Print() { | ||||
| 	total := struct { | ||||
| 		Count int | ||||
| 		Min   time.Duration | ||||
| 		Max   time.Duration | ||||
| 		Sum   time.Duration | ||||
| 		P90   time.Duration | ||||
| 		P95   time.Duration | ||||
| 		P99   time.Duration | ||||
| 	}{ | ||||
| 		Count: len(responses), | ||||
| 		Min:   responses[0].Time, | ||||
| 		Max:   responses[0].Time, | ||||
| 	if len(responses) == 0 { | ||||
| 		return | ||||
| 	} | ||||
| 	mergedResponses := make(map[string]types.Durations) | ||||
| 	var allDurations types.Durations | ||||
|  | ||||
| 	for _, response := range responses { | ||||
| 		if response.Time < total.Min { | ||||
| 			total.Min = response.Time | ||||
| 		} | ||||
| 		if response.Time > total.Max { | ||||
| 			total.Max = response.Time | ||||
| 		} | ||||
| 		total.Sum += response.Time | ||||
| 	mergedResponses := make(map[string]types.Durations) | ||||
|  | ||||
| 	totalDurations := make(types.Durations, len(responses)) | ||||
| 	var totalSum time.Duration | ||||
| 	totalCount := len(responses) | ||||
|  | ||||
| 	for i, response := range responses { | ||||
| 		totalSum += response.Time | ||||
| 		totalDurations[i] = response.Time | ||||
|  | ||||
| 		mergedResponses[response.Response] = append( | ||||
| 			mergedResponses[response.Response], | ||||
| 			response.Time, | ||||
| 		) | ||||
| 		allDurations = append(allDurations, response.Time) | ||||
| 	} | ||||
| 	allDurations.Sort() | ||||
| 	allDurationsLenAsFloat := float64(len(allDurations) - 1) | ||||
| 	total.P90 = allDurations[int(0.90*allDurationsLenAsFloat)] | ||||
| 	total.P95 = allDurations[int(0.95*allDurationsLenAsFloat)] | ||||
| 	total.P99 = allDurations[int(0.99*allDurationsLenAsFloat)] | ||||
|  | ||||
| 	t := table.NewWriter() | ||||
| 	t.SetOutputMirror(os.Stdout) | ||||
| @@ -93,15 +76,18 @@ func (responses Responses) Print() { | ||||
| 	} | ||||
|  | ||||
| 	if len(mergedResponses) > 1 { | ||||
| 		totalDurations.Sort() | ||||
| 		allDurationsLenAsFloat := float64(len(totalDurations) - 1) | ||||
|  | ||||
| 		t.AppendRow(table.Row{ | ||||
| 			"Total", | ||||
| 			total.Count, | ||||
| 			utils.DurationRoundBy(total.Min, roundPrecision), | ||||
| 			utils.DurationRoundBy(total.Max, roundPrecision), | ||||
| 			utils.DurationRoundBy(total.Sum/time.Duration(total.Count), roundPrecision), // Average | ||||
| 			utils.DurationRoundBy(total.P90, roundPrecision), | ||||
| 			utils.DurationRoundBy(total.P95, roundPrecision), | ||||
| 			utils.DurationRoundBy(total.P99, roundPrecision), | ||||
| 			totalCount, | ||||
| 			utils.DurationRoundBy(totalDurations[0], roundPrecision), | ||||
| 			utils.DurationRoundBy(totalDurations[len(totalDurations)-1], roundPrecision), | ||||
| 			utils.DurationRoundBy(totalSum/time.Duration(totalCount), roundPrecision), // Average | ||||
| 			utils.DurationRoundBy(totalDurations[int(0.90*allDurationsLenAsFloat)], roundPrecision), | ||||
| 			utils.DurationRoundBy(totalDurations[int(0.95*allDurationsLenAsFloat)], roundPrecision), | ||||
| 			utils.DurationRoundBy(totalDurations[int(0.99*allDurationsLenAsFloat)], roundPrecision), | ||||
| 		}) | ||||
| 	} | ||||
| 	t.Render() | ||||
|   | ||||
| @@ -66,7 +66,7 @@ func releaseDodos( | ||||
| 		streamWG            sync.WaitGroup | ||||
| 		requestCountPerDodo uint | ||||
| 		dodosCount          = requestConfig.GetValidDodosCountForRequests() | ||||
| 		responses           = make([][]*Response, dodosCount) | ||||
| 		responses           = make([][]Response, dodosCount) | ||||
| 		increase            = make(chan int64, requestConfig.RequestCount) | ||||
| 	) | ||||
|  | ||||
| @@ -123,7 +123,7 @@ func sendRequestByCount( | ||||
| 	request *Request, | ||||
| 	timeout time.Duration, | ||||
| 	requestCount uint, | ||||
| 	responseData *[]*Response, | ||||
| 	responseData *[]Response, | ||||
| 	increase chan<- int64, | ||||
| 	wg *sync.WaitGroup, | ||||
| ) { | ||||
| @@ -146,7 +146,7 @@ func sendRequestByCount( | ||||
| 				if err == types.ErrInterrupt { | ||||
| 					return | ||||
| 				} | ||||
| 				*responseData = append(*responseData, &Response{ | ||||
| 				*responseData = append(*responseData, Response{ | ||||
| 					Response: err.Error(), | ||||
| 					Time:     completedTime, | ||||
| 				}) | ||||
| @@ -154,7 +154,7 @@ func sendRequestByCount( | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			*responseData = append(*responseData, &Response{ | ||||
| 			*responseData = append(*responseData, Response{ | ||||
| 				Response: strconv.Itoa(response.StatusCode()), | ||||
| 				Time:     completedTime, | ||||
| 			}) | ||||
| @@ -170,7 +170,7 @@ func sendRequest( | ||||
| 	ctx context.Context, | ||||
| 	request *Request, | ||||
| 	timeout time.Duration, | ||||
| 	responseData *[]*Response, | ||||
| 	responseData *[]Response, | ||||
| 	increase chan<- int64, | ||||
| 	wg *sync.WaitGroup, | ||||
| ) { | ||||
| @@ -193,7 +193,7 @@ func sendRequest( | ||||
| 				if err == types.ErrInterrupt { | ||||
| 					return | ||||
| 				} | ||||
| 				*responseData = append(*responseData, &Response{ | ||||
| 				*responseData = append(*responseData, Response{ | ||||
| 					Response: err.Error(), | ||||
| 					Time:     completedTime, | ||||
| 				}) | ||||
| @@ -201,7 +201,7 @@ func sendRequest( | ||||
| 				return | ||||
| 			} | ||||
|  | ||||
| 			*responseData = append(*responseData, &Response{ | ||||
| 			*responseData = append(*responseData, Response{ | ||||
| 				Response: strconv.Itoa(response.StatusCode()), | ||||
| 				Time:     completedTime, | ||||
| 			}) | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package types | ||||
|  | ||||
| import ( | ||||
| 	"slices" | ||||
| 	"sort" | ||||
| 	"time" | ||||
| ) | ||||
| @@ -14,9 +15,7 @@ func (d Durations) Sort(ascending ...bool) { | ||||
| 			return d[i] > d[j] | ||||
| 		}) | ||||
| 	} else { // Otherwise, sort in ascending order | ||||
| 		sort.Slice(d, func(i, j int) bool { | ||||
| 			return d[i] < d[j] | ||||
| 		}) | ||||
| 		slices.Sort(d) | ||||
| 	} | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -2,8 +2,8 @@ package utils | ||||
|  | ||||
| import "math/rand" | ||||
|  | ||||
| func Flatten[T any](nested [][]*T) []*T { | ||||
| 	flattened := make([]*T, 0) | ||||
| func Flatten[T any](nested [][]T) []T { | ||||
| 	flattened := make([]T, 0) | ||||
| 	for _, n := range nested { | ||||
| 		flattened = append(flattened, n...) | ||||
| 	} | ||||
|   | ||||
| @@ -81,20 +81,28 @@ func (g *FuncMapGenerator) newFuncMap() *template.FuncMap { | ||||
| 		}, | ||||
| 		"strings_TrimPrefix": strings.TrimPrefix, | ||||
| 		"strings_TrimSuffix": strings.TrimSuffix, | ||||
| 		"strings_Join": func(sep string, values ...string) string { | ||||
| 			return strings.Join(values, sep) | ||||
| 		}, | ||||
|  | ||||
| 		// Dict | ||||
| 		"dict_Str": func(values ...any) map[string]string { | ||||
| 		"dict_Str": func(values ...string) map[string]string { | ||||
| 			dict := make(map[string]string) | ||||
| 			for i := 0; i < len(values); i += 2 { | ||||
| 				if i+1 < len(values) { | ||||
| 					key := values[i].(string) | ||||
| 					value := values[i+1].(string) | ||||
| 					key := values[i] | ||||
| 					value := values[i+1] | ||||
| 					dict[key] = value | ||||
| 				} | ||||
| 			} | ||||
| 			return dict | ||||
| 		}, | ||||
|  | ||||
| 		// Slice | ||||
| 		"slice_Str":  func(values ...string) []string { return values }, | ||||
| 		"slice_Int":  func(values ...int) []int { return values }, | ||||
| 		"slice_Uint": func(values ...uint) []uint { return values }, | ||||
|  | ||||
| 		// Body | ||||
| 		"body_FormData": func(kv map[string]string) string { | ||||
| 			var data bytes.Buffer | ||||
| @@ -383,12 +391,16 @@ func (g *FuncMapGenerator) newFuncMap() *template.FuncMap { | ||||
| 		"fakeit_Number":       g.localFaker.Number, | ||||
| 		"fakeit_Int":          g.localFaker.Int, | ||||
| 		"fakeit_IntN":         g.localFaker.IntN, | ||||
| 		"fakeit_IntRange":     g.localFaker.IntRange, | ||||
| 		"fakeit_RandomInt":    g.localFaker.RandomInt, | ||||
| 		"fakeit_Int8":         g.localFaker.Int8, | ||||
| 		"fakeit_Int16":        g.localFaker.Int16, | ||||
| 		"fakeit_Int32":        g.localFaker.Int32, | ||||
| 		"fakeit_Int64":        g.localFaker.Int64, | ||||
| 		"fakeit_Uint":         g.localFaker.Uint, | ||||
| 		"fakeit_UintN":        g.localFaker.UintN, | ||||
| 		"fakeit_UintRange":    g.localFaker.UintRange, | ||||
| 		"fakeit_RandomUint":   g.localFaker.RandomUint, | ||||
| 		"fakeit_Uint8":        g.localFaker.Uint8, | ||||
| 		"fakeit_Uint16":       g.localFaker.Uint16, | ||||
| 		"fakeit_Uint32":       g.localFaker.Uint32, | ||||
| @@ -404,8 +416,14 @@ func (g *FuncMapGenerator) newFuncMap() *template.FuncMap { | ||||
| 		"fakeit_DigitN":  g.localFaker.DigitN, | ||||
| 		"fakeit_Letter":  g.localFaker.Letter, | ||||
| 		"fakeit_LetterN": g.localFaker.LetterN, | ||||
| 		"fakeit_LetterNN": func(min, max uint) string { | ||||
| 			return g.localFaker.LetterN(g.localFaker.UintRange(min, max)) | ||||
| 		}, | ||||
| 		"fakeit_Lexify":   g.localFaker.Lexify, | ||||
| 		"fakeit_Numerify": g.localFaker.Numerify, | ||||
| 		"fakeit_RandomString": func(values ...string) string { | ||||
| 			return g.localFaker.RandomString(values) | ||||
| 		}, | ||||
|  | ||||
| 		// FakeIt / Celebrity | ||||
| 		"fakeit_CelebrityActor":    g.localFaker.CelebrityActor, | ||||
|   | ||||
		Reference in New Issue
	
	Block a user