mirror of
https://github.com/aykhans/dodo.git
synced 2025-06-03 04:46:41 +00:00
Compare commits
13 Commits
Author | SHA1 | Date | |
---|---|---|---|
5bb644d55f | |||
9152eefdc5 | |||
a8cd253c63 | |||
9aaf2db74d | |||
5c3e254e1e | |||
e5c681a22b | |||
79668e4ece | |||
f248c2af96 | |||
924bd819ee | |||
e567155eb1 | |||
23c74bdbb1 | |||
addf92df91 | |||
6aeda3706b |
@ -6,7 +6,7 @@ COPY go.mod go.sum ./
|
|||||||
RUN go mod download
|
RUN go mod download
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
RUN go build -ldflags "-s -w" -o dodo
|
RUN CGO_ENABLED=0 go build -ldflags "-s -w" -o dodo
|
||||||
|
|
||||||
FROM gcr.io/distroless/static-debian12:latest
|
FROM gcr.io/distroless/static-debian12:latest
|
||||||
|
|
||||||
|
216
README.md
216
README.md
@ -12,10 +12,11 @@
|
|||||||
- [Usage](#usage)
|
- [Usage](#usage)
|
||||||
- [1. CLI Usage](#1-cli-usage)
|
- [1. CLI Usage](#1-cli-usage)
|
||||||
- [2. Config File Usage](#2-config-file-usage)
|
- [2. Config File Usage](#2-config-file-usage)
|
||||||
- [2.1 JSON Example](#21-json-example)
|
- [2.1 YAML/YML Example](#21-yamlyml-example)
|
||||||
- [2.2 YAML/YML Example](#22-yamlyml-example)
|
- [2.2 JSON Example](#22-json-example)
|
||||||
- [3. CLI & Config File Combination](#3-cli--config-file-combination)
|
- [3. CLI & Config File Combination](#3-cli--config-file-combination)
|
||||||
- [Config Parameters Reference](#config-parameters-reference)
|
- [Config Parameters Reference](#config-parameters-reference)
|
||||||
|
- [Template Functions](#template-functions)
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@ -73,7 +74,84 @@ docker run --rm -i aykhans/dodo -u https://example.com -m GET -d 10 -r 1000 -o 1
|
|||||||
|
|
||||||
Send 1000 GET requests to https://example.com with 10 parallel dodos (threads), each with a timeout of 800 milliseconds, within a maximum duration of 250 seconds:
|
Send 1000 GET requests to https://example.com with 10 parallel dodos (threads), each with a timeout of 800 milliseconds, within a maximum duration of 250 seconds:
|
||||||
|
|
||||||
#### 2.1 JSON Example
|
#### 2.1 YAML/YML Example
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
method: "GET"
|
||||||
|
url: "https://example.com"
|
||||||
|
yes: false
|
||||||
|
timeout: "800ms"
|
||||||
|
dodos: 10
|
||||||
|
requests: 1000
|
||||||
|
duration: "250s"
|
||||||
|
skip_verify: false
|
||||||
|
|
||||||
|
params:
|
||||||
|
# A random value will be selected from the list for first "key1" param on each request
|
||||||
|
# And always "value" for second "key1" param on each request
|
||||||
|
# e.g. "?key1=value2&key1=value"
|
||||||
|
- key1: ["value1", "value2", "value3", "value4"]
|
||||||
|
- key1: "value"
|
||||||
|
|
||||||
|
# A random value will be selected from the list for param "key2" on each request
|
||||||
|
# e.g. "?key2=value2"
|
||||||
|
- key2: ["value1", "value2"]
|
||||||
|
|
||||||
|
headers:
|
||||||
|
# A random value will be selected from the list for first "key1" header on each request
|
||||||
|
# And always "value" for second "key1" header on each request
|
||||||
|
# e.g. "key1: value3", "key1: value"
|
||||||
|
- key1: ["value1", "value2", "value3", "value4"]
|
||||||
|
- key1: "value"
|
||||||
|
|
||||||
|
# A random value will be selected from the list for header "key2" on each request
|
||||||
|
# e.g. "key2: value2"
|
||||||
|
- key2: ["value1", "value2"]
|
||||||
|
|
||||||
|
cookies:
|
||||||
|
# A random value will be selected from the list for first "key1" cookie on each request
|
||||||
|
# And always "value" for second "key1" cookie on each request
|
||||||
|
# e.g. "key1=value4; key1=value"
|
||||||
|
- key1: ["value1", "value2", "value3", "value4"]
|
||||||
|
- key1: "value"
|
||||||
|
|
||||||
|
# A random value will be selected from the list for cookie "key2" on each request
|
||||||
|
# e.g. "key2=value1"
|
||||||
|
- key2: ["value1", "value2"]
|
||||||
|
|
||||||
|
body: "body-text"
|
||||||
|
# OR
|
||||||
|
# A random body value will be selected from the list for each request
|
||||||
|
body:
|
||||||
|
- "body-text1"
|
||||||
|
- "body-text2"
|
||||||
|
- "body-text3"
|
||||||
|
|
||||||
|
proxy: "http://example.com:8080"
|
||||||
|
# OR
|
||||||
|
# A random proxy will be selected from the list for each request
|
||||||
|
proxy:
|
||||||
|
- "http://example.com:8080"
|
||||||
|
- "http://username:password@example.com:8080"
|
||||||
|
- "socks5://example.com:8080"
|
||||||
|
- "socks5h://example.com:8080"
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh
|
||||||
|
dodo -f /path/config.yaml
|
||||||
|
# OR
|
||||||
|
dodo -f https://example.com/config.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
With Docker:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker run --rm -i -v /path/to/config.yaml:/config.yaml aykhans/dodo -f /config.yaml
|
||||||
|
# OR
|
||||||
|
docker run --rm -i aykhans/dodo -f https://example.com/config.yaml
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 2.2 JSON Example
|
||||||
|
|
||||||
```jsonc
|
```jsonc
|
||||||
{
|
{
|
||||||
@ -84,6 +162,7 @@ Send 1000 GET requests to https://example.com with 10 parallel dodos (threads),
|
|||||||
"dodos": 10,
|
"dodos": 10,
|
||||||
"requests": 1000,
|
"requests": 1000,
|
||||||
"duration": "250s",
|
"duration": "250s",
|
||||||
|
"skip_verify": false,
|
||||||
|
|
||||||
"params": [
|
"params": [
|
||||||
// A random value will be selected from the list for first "key1" param on each request
|
// A random value will be selected from the list for first "key1" param on each request
|
||||||
@ -152,82 +231,6 @@ docker run --rm -i -v /path/to/config.json:/config.json aykhans/dodo
|
|||||||
docker run --rm -i aykhans/dodo -f https://example.com/config.json
|
docker run --rm -i aykhans/dodo -f https://example.com/config.json
|
||||||
```
|
```
|
||||||
|
|
||||||
#### 2.2 YAML/YML Example
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
method: "GET"
|
|
||||||
url: "https://example.com"
|
|
||||||
yes: false
|
|
||||||
timeout: "800ms"
|
|
||||||
dodos: 10
|
|
||||||
requests: 1000
|
|
||||||
duration: "250s"
|
|
||||||
|
|
||||||
params:
|
|
||||||
# A random value will be selected from the list for first "key1" param on each request
|
|
||||||
# And always "value" for second "key1" param on each request
|
|
||||||
# e.g. "?key1=value2&key1=value"
|
|
||||||
- key1: ["value1", "value2", "value3", "value4"]
|
|
||||||
- key1: "value"
|
|
||||||
|
|
||||||
# A random value will be selected from the list for param "key2" on each request
|
|
||||||
# e.g. "?key2=value2"
|
|
||||||
- key2: ["value1", "value2"]
|
|
||||||
|
|
||||||
headers:
|
|
||||||
# A random value will be selected from the list for first "key1" header on each request
|
|
||||||
# And always "value" for second "key1" header on each request
|
|
||||||
# e.g. "key1: value3", "key1: value"
|
|
||||||
- key1: ["value1", "value2", "value3", "value4"]
|
|
||||||
- key1: "value"
|
|
||||||
|
|
||||||
# A random value will be selected from the list for header "key2" on each request
|
|
||||||
# e.g. "key2: value2"
|
|
||||||
- key2: ["value1", "value2"]
|
|
||||||
|
|
||||||
cookies:
|
|
||||||
# A random value will be selected from the list for first "key1" cookie on each request
|
|
||||||
# And always "value" for second "key1" cookie on each request
|
|
||||||
# e.g. "key1=value4; key1=value"
|
|
||||||
- key1: ["value1", "value2", "value3", "value4"]
|
|
||||||
- key1: "value"
|
|
||||||
|
|
||||||
# A random value will be selected from the list for cookie "key2" on each request
|
|
||||||
# e.g. "key2=value1"
|
|
||||||
- key2: ["value1", "value2"]
|
|
||||||
|
|
||||||
body: "body-text"
|
|
||||||
# OR
|
|
||||||
# A random body value will be selected from the list for each request
|
|
||||||
body:
|
|
||||||
- "body-text1"
|
|
||||||
- "body-text2"
|
|
||||||
- "body-text3"
|
|
||||||
|
|
||||||
proxy: "http://example.com:8080"
|
|
||||||
# OR
|
|
||||||
# A random proxy will be selected from the list for each request
|
|
||||||
proxy:
|
|
||||||
- "http://example.com:8080"
|
|
||||||
- "http://username:password@example.com:8080"
|
|
||||||
- "socks5://example.com:8080"
|
|
||||||
- "socks5h://example.com:8080"
|
|
||||||
```
|
|
||||||
|
|
||||||
```sh
|
|
||||||
dodo -f /path/config.yaml
|
|
||||||
# OR
|
|
||||||
dodo -f https://example.com/config.yaml
|
|
||||||
```
|
|
||||||
|
|
||||||
With Docker:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
docker run --rm -i -v /path/to/config.yaml:/config.yaml aykhans/dodo -f /config.yaml
|
|
||||||
# OR
|
|
||||||
docker run --rm -i aykhans/dodo -f https://example.com/config.yaml
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. CLI & Config File Combination
|
### 3. CLI & Config File Combination
|
||||||
|
|
||||||
CLI arguments override config file values:
|
CLI arguments override config file values:
|
||||||
@ -248,7 +251,7 @@ If `Headers`, `Params`, `Cookies`, `Body`, or `Proxy` fields have multiple value
|
|||||||
|
|
||||||
| Parameter | config file | CLI Flag | CLI Short Flag | Type | Description | Default |
|
| Parameter | config file | CLI Flag | CLI Short Flag | Type | Description | Default |
|
||||||
| --------------- | ----------- | ------------ | -------------- | ------------------------------ | ----------------------------------------------------------- | ------- |
|
| --------------- | ----------- | ------------ | -------------- | ------------------------------ | ----------------------------------------------------------- | ------- |
|
||||||
| Config file | - | -config-file | -f | String | Path to local config file or http(s) URL of the config file | - |
|
| Config file | | -config-file | -f | String | Path to local config file or http(s) URL of the config file | - |
|
||||||
| Yes | yes | -yes | -y | Boolean | Answer yes to all questions | false |
|
| Yes | yes | -yes | -y | Boolean | Answer yes to all questions | false |
|
||||||
| URL | url | -url | -u | String | URL to send the request to | - |
|
| URL | url | -url | -u | String | URL to send the request to | - |
|
||||||
| Method | method | -method | -m | String | HTTP method | GET |
|
| Method | method | -method | -m | String | HTTP method | GET |
|
||||||
@ -261,3 +264,56 @@ If `Headers`, `Params`, `Cookies`, `Body`, or `Proxy` fields have multiple value
|
|||||||
| Cookies | cookies | -cookie | -c | [{String: String OR [String]}] | Request cookies | - |
|
| Cookies | cookies | -cookie | -c | [{String: String OR [String]}] | Request cookies | - |
|
||||||
| Body | body | -body | -b | String OR [String] | Request body or list of request bodies | - |
|
| Body | body | -body | -b | String OR [String] | Request body or list of request bodies | - |
|
||||||
| Proxy | proxies | -proxy | -x | String OR [String] | Proxy URL or list of proxy URLs | - |
|
| Proxy | proxies | -proxy | -x | String OR [String] | Proxy URL or list of proxy URLs | - |
|
||||||
|
| Skip Verify | skip_verify | -skip-verify | | Boolean | Skip SSL/TLS certificate verification | false |
|
||||||
|
|
||||||
|
## Template Functions
|
||||||
|
|
||||||
|
Dodo supports template functions in `Headers`, `Params`, `Cookies`, and `Body` fields. These functions allow you to generate dynamic values for each request.
|
||||||
|
|
||||||
|
You can use Go template syntax to include dynamic values in your requests. Here's how to use template functions:
|
||||||
|
|
||||||
|
In CLI config:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
dodo -u https://example.com -r 1 \
|
||||||
|
-header "User-Agent:{{ fakeit_UserAgent }}" \ # e.g. "User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko)"
|
||||||
|
-param "username={{ strings_ToUpper fakeit_Username }}" \ # e.g. "username=JOHN BOB"
|
||||||
|
-cookie "token={{ fakeit_Password true true true true true 10 }}" \ # e.g. token=1234567890abcdef1234567890abcdef
|
||||||
|
-body '{"email":"{{ fakeit_Email }}", "password":"{{ fakeit_Password true true true true true 10 }}"}' # e.g. {"email":"john.doe@example.com", "password":"12rw4d-78d"}
|
||||||
|
```
|
||||||
|
|
||||||
|
In YAML/YML config:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
headers:
|
||||||
|
- User-Agent: "{{ fakeit_UserAgent }}" # e.g. "User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko)"
|
||||||
|
- "Random-Header-{{fakeit_FirstName}}": "static_value" # e.g. "Random-Header-John: static_value"
|
||||||
|
|
||||||
|
cookies:
|
||||||
|
- token: "Bearer {{ fakeit_UUID }}" # e.g. "token=Bearer 1234567890abcdef1234567890abcdef"
|
||||||
|
|
||||||
|
params:
|
||||||
|
- id: "{{ fakeit_Uint }}" # e.g. "id=1234567890"
|
||||||
|
- username: "{{ fakeit_Username }}" # e.g. "username=John Doe"
|
||||||
|
|
||||||
|
body:
|
||||||
|
- '{ "username": "{{ fakeit_Username }}", "password": "{{ fakeit_Password }}" }' # e.g. { "username": "john.doe", "password": "password123" }
|
||||||
|
- '{ "email": "{{ fakeit_Email }}", "phone": "{{ fakeit_Phone }}" }' # e.g. { "email": "john.doe@example.com", "phone": "1234567890" }
|
||||||
|
- '{{ body_FormData (dict_Str "username" fakeit_Username "password" "secret123") }}' # Creates multipart form data for form submissions, automatically sets the appropriate Content-Type header.
|
||||||
|
```
|
||||||
|
|
||||||
|
In JSON config:
|
||||||
|
|
||||||
|
```jsonc
|
||||||
|
{
|
||||||
|
"headers": [
|
||||||
|
{ "User-Agent": "{{ fakeit_UserAgent }}" }, // e.g. "User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko)"
|
||||||
|
],
|
||||||
|
"body": [
|
||||||
|
"{ \"username\": \"{{ strings_RemoveSpaces fakeit_Username }}\", \"password\": \"{{ fakeit_Password }}\" }", // e.g. { "username": "johndoe", "password": "password123" }
|
||||||
|
"{{ body_FormData (dict_Str \"username\" fakeit_Username \"password\" \"12345\") }}", // Creates multipart form data for form submissions, automatically sets the appropriate Content-Type header.
|
||||||
|
],
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
For the full list of template functions over 200 functions, refer to the `NewFuncMap` function in `utils/templates.go`.
|
||||||
|
@ -20,11 +20,19 @@ vars:
|
|||||||
tasks:
|
tasks:
|
||||||
run: go run main.go
|
run: go run main.go
|
||||||
|
|
||||||
|
ftl:
|
||||||
|
cmds:
|
||||||
|
- task: fmt
|
||||||
|
- task: tidy
|
||||||
|
- task: lint
|
||||||
|
|
||||||
fmt: gofmt -w -d .
|
fmt: gofmt -w -d .
|
||||||
|
|
||||||
|
tidy: go mod tidy
|
||||||
|
|
||||||
lint: golangci-lint run
|
lint: golangci-lint run
|
||||||
|
|
||||||
build: go build -ldflags "-s -w" -o "dodo"
|
build: CGO_ENABLED=0 go build -ldflags "-s -w" -o "dodo"
|
||||||
|
|
||||||
build-all:
|
build-all:
|
||||||
silent: true
|
silent: true
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
"dodos": 8,
|
"dodos": 8,
|
||||||
"requests": 1000,
|
"requests": 1000,
|
||||||
"duration": "10s",
|
"duration": "10s",
|
||||||
|
"skip_verify": false,
|
||||||
|
|
||||||
"params": [
|
"params": [
|
||||||
{ "key1": ["value1", "value2", "value3", "value4"] },
|
{ "key1": ["value1", "value2", "value3", "value4"] },
|
||||||
|
@ -5,6 +5,7 @@ timeout: "5s"
|
|||||||
dodos: 8
|
dodos: 8
|
||||||
requests: 1000
|
requests: 1000
|
||||||
duration: "10s"
|
duration: "10s"
|
||||||
|
skip_verify: false
|
||||||
|
|
||||||
params:
|
params:
|
||||||
- key1: ["value1", "value2", "value3", "value4"]
|
- key1: ["value1", "value2", "value3", "value4"]
|
||||||
|
@ -31,7 +31,7 @@ Usage with all flags:
|
|||||||
-p "param1=value1" -param "param2=value2" \
|
-p "param1=value1" -param "param2=value2" \
|
||||||
-c "cookie1=value1" -cookie "cookie2=value2" \
|
-c "cookie1=value1" -cookie "cookie2=value2" \
|
||||||
-x "http://proxy.example.com:8080" -proxy "socks5://proxy2.example.com:8080" \
|
-x "http://proxy.example.com:8080" -proxy "socks5://proxy2.example.com:8080" \
|
||||||
-y
|
-skip-verify -y
|
||||||
|
|
||||||
Flags:
|
Flags:
|
||||||
-h, -help help for dodo
|
-h, -help help for dodo
|
||||||
@ -48,7 +48,8 @@ Flags:
|
|||||||
-p, -param [string] Parameter for the request (e.g. "key1=value1")
|
-p, -param [string] Parameter for the request (e.g. "key1=value1")
|
||||||
-H, -header [string] Header for the request (e.g. "key1:value1")
|
-H, -header [string] Header for the request (e.g. "key1:value1")
|
||||||
-c, -cookie [string] Cookie for the request (e.g. "key1=value1")
|
-c, -cookie [string] Cookie for the request (e.g. "key1=value1")
|
||||||
-x, -proxy [string] Proxy for the request (e.g. "http://proxy.example.com:8080")`
|
-x, -proxy [string] Proxy for the request (e.g. "http://proxy.example.com:8080")
|
||||||
|
-skip-verify bool Skip SSL/TLS certificate verification (default %v)`
|
||||||
|
|
||||||
func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
||||||
flag.Usage = func() {
|
flag.Usage = func() {
|
||||||
@ -58,6 +59,7 @@ func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
|||||||
DefaultDodosCount,
|
DefaultDodosCount,
|
||||||
DefaultTimeout,
|
DefaultTimeout,
|
||||||
DefaultMethod,
|
DefaultMethod,
|
||||||
|
DefaultSkipVerify,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,6 +67,7 @@ func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
|||||||
version = false
|
version = false
|
||||||
configFile = ""
|
configFile = ""
|
||||||
yes = false
|
yes = false
|
||||||
|
skipVerify = false
|
||||||
method = ""
|
method = ""
|
||||||
url types.RequestURL
|
url types.RequestURL
|
||||||
dodosCount = uint(0)
|
dodosCount = uint(0)
|
||||||
@ -83,6 +86,8 @@ func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
|||||||
flag.BoolVar(&yes, "yes", false, "Answer yes to all questions")
|
flag.BoolVar(&yes, "yes", false, "Answer yes to all questions")
|
||||||
flag.BoolVar(&yes, "y", false, "Answer yes to all questions")
|
flag.BoolVar(&yes, "y", false, "Answer yes to all questions")
|
||||||
|
|
||||||
|
flag.BoolVar(&skipVerify, "skip-verify", false, "Skip SSL/TLS certificate verification")
|
||||||
|
|
||||||
flag.StringVar(&method, "method", "", "HTTP Method")
|
flag.StringVar(&method, "method", "", "HTTP Method")
|
||||||
flag.StringVar(&method, "m", "", "HTTP Method")
|
flag.StringVar(&method, "m", "", "HTTP Method")
|
||||||
|
|
||||||
@ -149,6 +154,8 @@ func (config *Config) ReadCLI() (types.ConfigFile, error) {
|
|||||||
config.Timeout = &types.Timeout{Duration: timeout}
|
config.Timeout = &types.Timeout{Duration: timeout}
|
||||||
case "yes", "y":
|
case "yes", "y":
|
||||||
config.Yes = utils.ToPtr(yes)
|
config.Yes = utils.ToPtr(yes)
|
||||||
|
case "skip-verify":
|
||||||
|
config.SkipVerify = utils.ToPtr(skipVerify)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
109
config/config.go
109
config/config.go
@ -1,12 +1,15 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"slices"
|
"slices"
|
||||||
"strings"
|
"strings"
|
||||||
|
"text/template"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/aykhans/dodo/types"
|
"github.com/aykhans/dodo/types"
|
||||||
@ -15,7 +18,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
VERSION string = "0.6.4"
|
VERSION string = "0.7.0"
|
||||||
DefaultUserAgent string = "Dodo/" + VERSION
|
DefaultUserAgent string = "Dodo/" + VERSION
|
||||||
DefaultMethod string = "GET"
|
DefaultMethod string = "GET"
|
||||||
DefaultTimeout time.Duration = time.Second * 10
|
DefaultTimeout time.Duration = time.Second * 10
|
||||||
@ -23,6 +26,7 @@ const (
|
|||||||
DefaultRequestCount uint = 0
|
DefaultRequestCount uint = 0
|
||||||
DefaultDuration time.Duration = 0
|
DefaultDuration time.Duration = 0
|
||||||
DefaultYes bool = false
|
DefaultYes bool = false
|
||||||
|
DefaultSkipVerify bool = false
|
||||||
)
|
)
|
||||||
|
|
||||||
var SupportedProxySchemes []string = []string{"http", "socks5", "socks5h"}
|
var SupportedProxySchemes []string = []string{"http", "socks5", "socks5h"}
|
||||||
@ -35,6 +39,7 @@ type RequestConfig struct {
|
|||||||
RequestCount uint
|
RequestCount uint
|
||||||
Duration time.Duration
|
Duration time.Duration
|
||||||
Yes bool
|
Yes bool
|
||||||
|
SkipVerify bool
|
||||||
Params types.Params
|
Params types.Params
|
||||||
Headers types.Headers
|
Headers types.Headers
|
||||||
Cookies types.Cookies
|
Cookies types.Cookies
|
||||||
@ -51,6 +56,7 @@ func NewRequestConfig(conf *Config) *RequestConfig {
|
|||||||
RequestCount: *conf.RequestCount,
|
RequestCount: *conf.RequestCount,
|
||||||
Duration: conf.Duration.Duration,
|
Duration: conf.Duration.Duration,
|
||||||
Yes: *conf.Yes,
|
Yes: *conf.Yes,
|
||||||
|
SkipVerify: *conf.SkipVerify,
|
||||||
Params: conf.Params,
|
Params: conf.Params,
|
||||||
Headers: conf.Headers,
|
Headers: conf.Headers,
|
||||||
Cookies: conf.Cookies,
|
Cookies: conf.Cookies,
|
||||||
@ -122,6 +128,8 @@ func (rc *RequestConfig) Print() {
|
|||||||
t.AppendRow(table.Row{"Proxy", rc.Proxies.String()})
|
t.AppendRow(table.Row{"Proxy", rc.Proxies.String()})
|
||||||
t.AppendSeparator()
|
t.AppendSeparator()
|
||||||
t.AppendRow(table.Row{"Body", rc.Body.String()})
|
t.AppendRow(table.Row{"Body", rc.Body.String()})
|
||||||
|
t.AppendSeparator()
|
||||||
|
t.AppendRow(table.Row{"Skip Verify", rc.SkipVerify})
|
||||||
|
|
||||||
t.Render()
|
t.Render()
|
||||||
}
|
}
|
||||||
@ -134,6 +142,7 @@ type Config struct {
|
|||||||
RequestCount *uint `json:"requests" yaml:"requests"`
|
RequestCount *uint `json:"requests" yaml:"requests"`
|
||||||
Duration *types.Duration `json:"duration" yaml:"duration"`
|
Duration *types.Duration `json:"duration" yaml:"duration"`
|
||||||
Yes *bool `json:"yes" yaml:"yes"`
|
Yes *bool `json:"yes" yaml:"yes"`
|
||||||
|
SkipVerify *bool `json:"skip_verify" yaml:"skip_verify"`
|
||||||
Params types.Params `json:"params" yaml:"params"`
|
Params types.Params `json:"params" yaml:"params"`
|
||||||
Headers types.Headers `json:"headers" yaml:"headers"`
|
Headers types.Headers `json:"headers" yaml:"headers"`
|
||||||
Cookies types.Cookies `json:"cookies" yaml:"cookies"`
|
Cookies types.Cookies `json:"cookies" yaml:"cookies"`
|
||||||
@ -195,6 +204,98 @@ func (config *Config) Validate() []error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
funcMap := *utils.NewFuncMapGenerator(
|
||||||
|
rand.New(
|
||||||
|
rand.NewSource(
|
||||||
|
time.Now().UnixNano(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
).GetFuncMap()
|
||||||
|
|
||||||
|
for _, header := range config.Headers {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(header.Key)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("header key (%s) parse error: %v", header.Key, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("header key (%s) parse error: %v", header.Key, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, value := range header.Value {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(value)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("header value (%s) parse error: %v", value, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("header value (%s) parse error: %v", value, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, cookie := range config.Cookies {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(cookie.Key)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("cookie key (%s) parse error: %v", cookie.Key, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("cookie key (%s) parse error: %v", cookie.Key, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, value := range cookie.Value {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(value)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("cookie value (%s) parse error: %v", value, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("cookie value (%s) parse error: %v", value, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, param := range config.Params {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(param.Key)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("param key (%s) parse error: %v", param.Key, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("param key (%s) parse error: %v", param.Key, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, value := range param.Value {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(value)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("param value (%s) parse error: %v", value, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("param value (%s) parse error: %v", value, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, body := range config.Body {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(body)
|
||||||
|
if err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("body (%s) parse error: %v", body, err))
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err = t.Execute(&buf, nil); err != nil {
|
||||||
|
errs = append(errs, fmt.Errorf("body (%s) parse error: %v", body, err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return errs
|
return errs
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -220,6 +321,9 @@ func (config *Config) MergeConfig(newConfig *Config) {
|
|||||||
if newConfig.Yes != nil {
|
if newConfig.Yes != nil {
|
||||||
config.Yes = newConfig.Yes
|
config.Yes = newConfig.Yes
|
||||||
}
|
}
|
||||||
|
if newConfig.SkipVerify != nil {
|
||||||
|
config.SkipVerify = newConfig.SkipVerify
|
||||||
|
}
|
||||||
if len(newConfig.Params) != 0 {
|
if len(newConfig.Params) != 0 {
|
||||||
config.Params = newConfig.Params
|
config.Params = newConfig.Params
|
||||||
}
|
}
|
||||||
@ -256,5 +360,8 @@ func (config *Config) SetDefaults() {
|
|||||||
if config.Yes == nil {
|
if config.Yes == nil {
|
||||||
config.Yes = utils.ToPtr(DefaultYes)
|
config.Yes = utils.ToPtr(DefaultYes)
|
||||||
}
|
}
|
||||||
|
if config.SkipVerify == nil {
|
||||||
|
config.SkipVerify = utils.ToPtr(DefaultSkipVerify)
|
||||||
|
}
|
||||||
config.Headers.SetIfNotExists("User-Agent", DefaultUserAgent)
|
config.Headers.SetIfNotExists("User-Agent", DefaultUserAgent)
|
||||||
}
|
}
|
||||||
|
1
go.mod
1
go.mod
@ -3,6 +3,7 @@ module github.com/aykhans/dodo
|
|||||||
go 1.24.2
|
go 1.24.2
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
github.com/brianvoe/gofakeit/v7 v7.2.1
|
||||||
github.com/jedib0t/go-pretty/v6 v6.6.7
|
github.com/jedib0t/go-pretty/v6 v6.6.7
|
||||||
github.com/valyala/fasthttp v1.62.0
|
github.com/valyala/fasthttp v1.62.0
|
||||||
gopkg.in/yaml.v3 v3.0.1
|
gopkg.in/yaml.v3 v3.0.1
|
||||||
|
2
go.sum
2
go.sum
@ -1,5 +1,7 @@
|
|||||||
github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA=
|
github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA=
|
||||||
github.com/andybalholm/brotli v1.1.1/go.mod h1:05ib4cKhjx3OQYUY22hTVd34Bc8upXjOLL2rKwwZBoA=
|
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/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/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 h1:m+LbHpm0aIAPLzLbMfn8dc3Ht8MW7lsSO4MPItz/Uuo=
|
||||||
|
@ -18,11 +18,12 @@ type ClientGeneratorFunc func() *fasthttp.HostClient
|
|||||||
// getClients initializes and returns a slice of fasthttp.HostClient based on the provided parameters.
|
// getClients initializes and returns a slice of fasthttp.HostClient based on the provided parameters.
|
||||||
// It can either return clients with proxies or a single client without proxies.
|
// It can either return clients with proxies or a single client without proxies.
|
||||||
func getClients(
|
func getClients(
|
||||||
ctx context.Context,
|
_ context.Context,
|
||||||
timeout time.Duration,
|
timeout time.Duration,
|
||||||
proxies []url.URL,
|
proxies []url.URL,
|
||||||
maxConns uint,
|
maxConns uint,
|
||||||
URL url.URL,
|
URL url.URL,
|
||||||
|
skipVerify bool,
|
||||||
) []*fasthttp.HostClient {
|
) []*fasthttp.HostClient {
|
||||||
isTLS := URL.Scheme == "https"
|
isTLS := URL.Scheme == "https"
|
||||||
|
|
||||||
@ -43,7 +44,7 @@ func getClients(
|
|||||||
MaxConns: int(maxConns),
|
MaxConns: int(maxConns),
|
||||||
IsTLS: isTLS,
|
IsTLS: isTLS,
|
||||||
TLSConfig: &tls.Config{
|
TLSConfig: &tls.Config{
|
||||||
InsecureSkipVerify: true,
|
InsecureSkipVerify: skipVerify,
|
||||||
},
|
},
|
||||||
Addr: addr,
|
Addr: addr,
|
||||||
Dial: dialFunc,
|
Dial: dialFunc,
|
||||||
@ -61,7 +62,7 @@ func getClients(
|
|||||||
MaxConns: int(maxConns),
|
MaxConns: int(maxConns),
|
||||||
IsTLS: isTLS,
|
IsTLS: isTLS,
|
||||||
TLSConfig: &tls.Config{
|
TLSConfig: &tls.Config{
|
||||||
InsecureSkipVerify: true,
|
InsecureSkipVerify: skipVerify,
|
||||||
},
|
},
|
||||||
Addr: URL.Host,
|
Addr: URL.Host,
|
||||||
MaxIdleConnDuration: timeout,
|
MaxIdleConnDuration: timeout,
|
||||||
@ -87,6 +88,11 @@ func getDialFunc(proxy *url.URL, timeout time.Duration) (fasthttp.DialFunc, erro
|
|||||||
default:
|
default:
|
||||||
return nil, errors.New("unsupported proxy scheme")
|
return nil, errors.New("unsupported proxy scheme")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if dialer == nil {
|
||||||
|
return nil, errors.New("internal error: proxy dialer is nil")
|
||||||
|
}
|
||||||
|
|
||||||
return dialer, nil
|
return dialer, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
package requests
|
package requests
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"text/template"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/aykhans/dodo/config"
|
"github.com/aykhans/dodo/config"
|
||||||
@ -21,6 +23,11 @@ type Request struct {
|
|||||||
getRequest RequestGeneratorFunc
|
getRequest RequestGeneratorFunc
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type keyValueGenerator struct {
|
||||||
|
key func() string
|
||||||
|
value func() string
|
||||||
|
}
|
||||||
|
|
||||||
// Send sends the HTTP request using the fasthttp client with a specified timeout.
|
// Send sends the HTTP request using the fasthttp client with a specified timeout.
|
||||||
// It returns the HTTP response or an error if the request fails or times out.
|
// It returns the HTTP response or an error if the request fails or times out.
|
||||||
func (r *Request) Send(ctx context.Context, timeout time.Duration) (*fasthttp.Response, error) {
|
func (r *Request) Send(ctx context.Context, timeout time.Duration) (*fasthttp.Response, error) {
|
||||||
@ -101,26 +108,28 @@ func getRequestGeneratorFunc(
|
|||||||
bodies []string,
|
bodies []string,
|
||||||
localRand *rand.Rand,
|
localRand *rand.Rand,
|
||||||
) RequestGeneratorFunc {
|
) RequestGeneratorFunc {
|
||||||
bodiesLen := len(bodies)
|
|
||||||
getBody := func() string { return "" }
|
|
||||||
if bodiesLen == 1 {
|
|
||||||
getBody = func() string { return bodies[0] }
|
|
||||||
} else if bodiesLen > 1 {
|
|
||||||
getBody = utils.RandomValueCycle(bodies, localRand)
|
|
||||||
}
|
|
||||||
|
|
||||||
getParams := getKeyValueGeneratorFunc(params, localRand)
|
getParams := getKeyValueGeneratorFunc(params, localRand)
|
||||||
getHeaders := getKeyValueGeneratorFunc(headers, localRand)
|
getHeaders := getKeyValueGeneratorFunc(headers, localRand)
|
||||||
getCookies := getKeyValueGeneratorFunc(cookies, localRand)
|
getCookies := getKeyValueGeneratorFunc(cookies, localRand)
|
||||||
|
getBody := getBodyValueFunc(bodies, utils.NewFuncMapGenerator(localRand), localRand)
|
||||||
|
|
||||||
return func() *fasthttp.Request {
|
return func() *fasthttp.Request {
|
||||||
|
body, contentType := getBody()
|
||||||
|
headers := getHeaders()
|
||||||
|
if contentType != "" {
|
||||||
|
headers = append(headers, types.KeyValue[string, string]{
|
||||||
|
Key: "Content-Type",
|
||||||
|
Value: contentType,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
return newFasthttpRequest(
|
return newFasthttpRequest(
|
||||||
URL,
|
URL,
|
||||||
getParams(),
|
getParams(),
|
||||||
getHeaders(),
|
headers,
|
||||||
getCookies(),
|
getCookies(),
|
||||||
method,
|
method,
|
||||||
getBody(),
|
body,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -199,49 +208,134 @@ func getKeyValueGeneratorFunc[
|
|||||||
keyValueSlice []types.KeyValue[string, []string],
|
keyValueSlice []types.KeyValue[string, []string],
|
||||||
localRand *rand.Rand,
|
localRand *rand.Rand,
|
||||||
) func() T {
|
) func() T {
|
||||||
getKeyValueSlice := []map[string]func() string{}
|
keyValueGenerators := make([]keyValueGenerator, len(keyValueSlice))
|
||||||
isRandom := false
|
|
||||||
|
|
||||||
for _, kv := range keyValueSlice {
|
funcMap := *utils.NewFuncMapGenerator(localRand).GetFuncMap()
|
||||||
valuesLen := len(kv.Value)
|
|
||||||
|
|
||||||
getValueFunc := func() string { return "" }
|
for i, kv := range keyValueSlice {
|
||||||
if valuesLen == 1 {
|
keyValueGenerators[i] = keyValueGenerator{
|
||||||
getValueFunc = func() string { return kv.Value[0] }
|
key: getKeyFunc(kv.Key, funcMap),
|
||||||
} else if valuesLen > 1 {
|
value: getValueFunc(kv.Value, funcMap, localRand),
|
||||||
getValueFunc = utils.RandomValueCycle(kv.Value, localRand)
|
}
|
||||||
isRandom = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getKeyValueSlice = append(
|
|
||||||
getKeyValueSlice,
|
|
||||||
map[string]func() string{kv.Key: getValueFunc},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if isRandom {
|
|
||||||
return func() T {
|
return func() T {
|
||||||
keyValues := make(T, len(getKeyValueSlice))
|
keyValues := make(T, len(keyValueGenerators))
|
||||||
for i, keyValue := range getKeyValueSlice {
|
for i, keyValue := range keyValueGenerators {
|
||||||
for key, value := range keyValue {
|
|
||||||
keyValues[i] = types.KeyValue[string, string]{
|
keyValues[i] = types.KeyValue[string, string]{
|
||||||
Key: key,
|
Key: keyValue.key(),
|
||||||
Value: value(),
|
Value: keyValue.value(),
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return keyValues
|
return keyValues
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
keyValues := make(T, len(getKeyValueSlice))
|
|
||||||
for i, keyValue := range getKeyValueSlice {
|
// getKeyFunc creates a function that processes a key string through Go's template engine.
|
||||||
for key, value := range keyValue {
|
// It takes a key string and a template.FuncMap containing the available template functions.
|
||||||
keyValues[i] = types.KeyValue[string, string]{
|
//
|
||||||
Key: key,
|
// The returned function, when called, will execute the template with the given key and return
|
||||||
Value: value(),
|
// the processed string result. If template parsing fails, the returned function will always
|
||||||
|
// return an empty string.
|
||||||
|
//
|
||||||
|
// This enables dynamic generation of keys that can include template directives and functions.
|
||||||
|
func getKeyFunc(key string, funcMap template.FuncMap) func() string {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(key)
|
||||||
|
if err != nil {
|
||||||
|
return func() string { return "" }
|
||||||
|
}
|
||||||
|
|
||||||
|
return func() string {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
_ = t.Execute(&buf, nil)
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// getValueFunc creates a function that randomly selects and processes a value from a slice of strings
|
||||||
|
// through Go's template engine.
|
||||||
|
//
|
||||||
|
// Parameters:
|
||||||
|
// - values: A slice of string templates that can contain template directives
|
||||||
|
// - funcMap: A template.FuncMap containing all available template functions
|
||||||
|
// - localRand: A random number generator for consistent randomization
|
||||||
|
//
|
||||||
|
// The returned function, when called, will:
|
||||||
|
// 1. Select a random template from the values slice
|
||||||
|
// 2. Execute the selected template
|
||||||
|
// 3. Return the processed string result
|
||||||
|
//
|
||||||
|
// If a selected template is nil (due to earlier parsing failure), the function will return an empty string.
|
||||||
|
// This enables dynamic generation of values with randomized selection from multiple templates.
|
||||||
|
func getValueFunc(
|
||||||
|
values []string,
|
||||||
|
funcMap template.FuncMap,
|
||||||
|
localRand *rand.Rand,
|
||||||
|
) func() string {
|
||||||
|
templates := make([]*template.Template, len(values))
|
||||||
|
|
||||||
|
for i, value := range values {
|
||||||
|
t, err := template.New("default").Funcs(funcMap).Parse(value)
|
||||||
|
if err != nil {
|
||||||
|
templates[i] = nil
|
||||||
|
}
|
||||||
|
templates[i] = t
|
||||||
|
}
|
||||||
|
|
||||||
|
randomTemplateFunc := utils.RandomValueCycle(templates, localRand)
|
||||||
|
|
||||||
|
return func() string {
|
||||||
|
if tmpl := randomTemplateFunc(); tmpl == nil {
|
||||||
|
return ""
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
_ = tmpl.Execute(&buf, nil)
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// getBodyValueFunc creates a function that randomly selects and processes a request body from a slice of templates.
|
||||||
|
// It returns a closure that generates both the body content and the appropriate Content-Type header value.
|
||||||
|
//
|
||||||
|
// Parameters:
|
||||||
|
// - values: A slice of string templates that can contain template directives for request bodies
|
||||||
|
// - funcMapGenerator: Provides template functions and content type information
|
||||||
|
// - localRand: A random number generator for consistent randomization
|
||||||
|
//
|
||||||
|
// The returned function, when called, will:
|
||||||
|
// 1. Select a random body template from the values slice
|
||||||
|
// 2. Execute the selected template with available template functions
|
||||||
|
// 3. Return both the processed body string and the appropriate Content-Type header value
|
||||||
|
//
|
||||||
|
// If the selected template is nil (due to earlier parsing failure), the function will return
|
||||||
|
// empty strings for both the body and Content-Type.
|
||||||
|
//
|
||||||
|
// This enables dynamic generation of request bodies with proper content type headers.
|
||||||
|
func getBodyValueFunc(
|
||||||
|
values []string,
|
||||||
|
funcMapGenerator *utils.FuncMapGenerator,
|
||||||
|
localRand *rand.Rand,
|
||||||
|
) func() (string, string) {
|
||||||
|
templates := make([]*template.Template, len(values))
|
||||||
|
|
||||||
|
for i, value := range values {
|
||||||
|
t, err := template.New("default").Funcs(*funcMapGenerator.GetFuncMap()).Parse(value)
|
||||||
|
if err != nil {
|
||||||
|
templates[i] = nil
|
||||||
|
}
|
||||||
|
templates[i] = t
|
||||||
|
}
|
||||||
|
|
||||||
|
randomTemplateFunc := utils.RandomValueCycle(templates, localRand)
|
||||||
|
|
||||||
|
return func() (string, string) {
|
||||||
|
if tmpl := randomTemplateFunc(); tmpl == nil {
|
||||||
|
return "", ""
|
||||||
|
} else {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
_ = tmpl.Execute(&buf, nil)
|
||||||
|
return buf.String(), funcMapGenerator.GetBodyDataHeader()
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
return func() T { return keyValues }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -32,6 +32,7 @@ func Run(ctx context.Context, requestConfig *config.RequestConfig) (Responses, e
|
|||||||
requestConfig.Proxies,
|
requestConfig.Proxies,
|
||||||
requestConfig.GetMaxConns(fasthttp.DefaultMaxConnsPerHost),
|
requestConfig.GetMaxConns(fasthttp.DefaultMaxConnsPerHost),
|
||||||
requestConfig.URL,
|
requestConfig.URL,
|
||||||
|
requestConfig.SkipVerify,
|
||||||
)
|
)
|
||||||
if clients == nil {
|
if clients == nil {
|
||||||
return nil, types.ErrInterrupt
|
return nil, types.ErrInterrupt
|
||||||
@ -71,7 +72,7 @@ func releaseDodos(
|
|||||||
|
|
||||||
wg.Add(int(dodosCount))
|
wg.Add(int(dodosCount))
|
||||||
streamWG.Add(1)
|
streamWG.Add(1)
|
||||||
streamCtx, streamCtxCancel := context.WithCancel(context.Background())
|
streamCtx, streamCtxCancel := context.WithCancel(ctx)
|
||||||
|
|
||||||
go streamProgress(streamCtx, &streamWG, requestConfig.RequestCount, "Dodos Working🔥", increase)
|
go streamProgress(streamCtx, &streamWG, requestConfig.RequestCount, "Dodos Working🔥", increase)
|
||||||
|
|
||||||
|
@ -66,7 +66,7 @@ func (body *Body) UnmarshalJSON(b []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (body *Body) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (body *Body) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var data any
|
var data any
|
||||||
if err := unmarshal(&data); err != nil {
|
if err := unmarshal(&data); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -99,7 +99,7 @@ func (cookies *Cookies) UnmarshalJSON(b []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cookies *Cookies) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (cookies *Cookies) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var raw []map[string]any
|
var raw []map[string]any
|
||||||
if err := unmarshal(&raw); err != nil {
|
if err := unmarshal(&raw); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -35,7 +35,7 @@ func (duration Duration) MarshalJSON() ([]byte, error) {
|
|||||||
return json.Marshal(duration.String())
|
return json.Marshal(duration.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (duration *Duration) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (duration *Duration) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var v any
|
var v any
|
||||||
if err := unmarshal(&v); err != nil {
|
if err := unmarshal(&v); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -108,7 +108,7 @@ func (headers *Headers) UnmarshalJSON(b []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (headers *Headers) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (headers *Headers) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var raw []map[string]any
|
var raw []map[string]any
|
||||||
if err := unmarshal(&raw); err != nil {
|
if err := unmarshal(&raw); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -99,7 +99,7 @@ func (params *Params) UnmarshalJSON(b []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (params *Params) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (params *Params) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var raw []map[string]any
|
var raw []map[string]any
|
||||||
if err := unmarshal(&raw); err != nil {
|
if err := unmarshal(&raw); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -75,7 +75,7 @@ func (proxies *Proxies) UnmarshalJSON(b []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (proxies *Proxies) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (proxies *Proxies) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var data any
|
var data any
|
||||||
if err := unmarshal(&data); err != nil {
|
if err := unmarshal(&data); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -25,7 +25,7 @@ func (requestURL *RequestURL) UnmarshalJSON(data []byte) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (requestURL *RequestURL) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (requestURL *RequestURL) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var urlStr string
|
var urlStr string
|
||||||
if err := unmarshal(&urlStr); err != nil {
|
if err := unmarshal(&urlStr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -35,7 +35,7 @@ func (timeout Timeout) MarshalJSON() ([]byte, error) {
|
|||||||
return json.Marshal(timeout.String())
|
return json.Marshal(timeout.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (timeout *Timeout) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
func (timeout *Timeout) UnmarshalYAML(unmarshal func(any) error) error {
|
||||||
var v any
|
var v any
|
||||||
if err := unmarshal(&v); err != nil {
|
if err := unmarshal(&v); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -10,32 +10,33 @@ func Flatten[T any](nested [][]*T) []*T {
|
|||||||
return flattened
|
return flattened
|
||||||
}
|
}
|
||||||
|
|
||||||
// RandomValueCycle returns a function that cycles through the provided slice of values
|
// RandomValueCycle returns a function that cycles through the provided values in a pseudo-random order.
|
||||||
// in a random order. Each call to the returned function will yield a value from the slice.
|
// Each value in the input slice will be returned before any value is repeated.
|
||||||
// The order of values is determined by the provided random number generator.
|
// If the input slice is empty, the returned function will always return the zero value of type T.
|
||||||
//
|
// If the input slice contains only one element, that element is always returned.
|
||||||
// The returned function will cycle through the values in a random order until all values
|
// This function is not thread-safe and should not be called concurrently.
|
||||||
// have been returned at least once. After all values have been returned, the function will
|
func RandomValueCycle[T any](values []T, localRand *rand.Rand) func() T {
|
||||||
// reset and start cycling through the values in a random order again.
|
switch valuesLen := len(values); valuesLen {
|
||||||
// The returned function isn't thread-safe and should be used in a single-threaded context.
|
case 0:
|
||||||
func RandomValueCycle[Value any](values []Value, localRand *rand.Rand) func() Value {
|
var zero T
|
||||||
var (
|
return func() T { return zero }
|
||||||
clientsCount = len(values)
|
case 1:
|
||||||
currentIndex = localRand.Intn(clientsCount)
|
return func() T { return values[0] }
|
||||||
stopIndex = currentIndex
|
default:
|
||||||
)
|
currentIndex := localRand.Intn(valuesLen)
|
||||||
|
stopIndex := currentIndex
|
||||||
return func() Value {
|
return func() T {
|
||||||
client := values[currentIndex]
|
value := values[currentIndex]
|
||||||
currentIndex++
|
currentIndex++
|
||||||
if currentIndex == clientsCount {
|
if currentIndex == valuesLen {
|
||||||
currentIndex = 0
|
currentIndex = 0
|
||||||
}
|
}
|
||||||
if currentIndex == stopIndex {
|
if currentIndex == stopIndex {
|
||||||
currentIndex = localRand.Intn(clientsCount)
|
currentIndex = localRand.Intn(valuesLen)
|
||||||
stopIndex = currentIndex
|
stopIndex = currentIndex
|
||||||
}
|
}
|
||||||
|
|
||||||
return client
|
return value
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
461
utils/templates.go
Normal file
461
utils/templates.go
Normal file
@ -0,0 +1,461 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"math/rand"
|
||||||
|
"mime/multipart"
|
||||||
|
"strings"
|
||||||
|
"text/template"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/brianvoe/gofakeit/v7"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FuncMapGenerator struct {
|
||||||
|
bodyDataHeader string
|
||||||
|
localFaker *gofakeit.Faker
|
||||||
|
funcMap *template.FuncMap
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewFuncMapGenerator(localRand *rand.Rand) *FuncMapGenerator {
|
||||||
|
f := &FuncMapGenerator{
|
||||||
|
localFaker: gofakeit.NewFaker(localRand, false),
|
||||||
|
}
|
||||||
|
f.funcMap = f.newFuncMap()
|
||||||
|
|
||||||
|
return f
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *FuncMapGenerator) GetBodyDataHeader() string {
|
||||||
|
tempHeader := g.bodyDataHeader
|
||||||
|
g.bodyDataHeader = ""
|
||||||
|
return tempHeader
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *FuncMapGenerator) GetFuncMap() *template.FuncMap {
|
||||||
|
return g.funcMap
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewFuncMap creates a template.FuncMap populated with string manipulation functions
|
||||||
|
// and data generation functions from gofakeit.
|
||||||
|
//
|
||||||
|
// It takes a random number generator that is used to initialize a localized faker
|
||||||
|
// instance, ensuring that random data generation is deterministic within a request context.
|
||||||
|
//
|
||||||
|
// All functions are prefixed to avoid naming conflicts:
|
||||||
|
// - String functions: "strings_*"
|
||||||
|
// - Dict functions: "dict_*"
|
||||||
|
// - Body functions: "body_*"
|
||||||
|
// - Data generation functions: "fakeit_*"
|
||||||
|
func (g *FuncMapGenerator) newFuncMap() *template.FuncMap {
|
||||||
|
return &template.FuncMap{
|
||||||
|
// Strings
|
||||||
|
"strings_ToUpper": strings.ToUpper,
|
||||||
|
"strings_ToLower": strings.ToLower,
|
||||||
|
"strings_RemoveSpaces": func(s string) string { return strings.ReplaceAll(s, " ", "") },
|
||||||
|
"strings_Replace": strings.Replace,
|
||||||
|
"strings_ToDate": func(dateString string) time.Time {
|
||||||
|
date, err := time.Parse("2006-01-02", dateString)
|
||||||
|
if err != nil {
|
||||||
|
return time.Now()
|
||||||
|
}
|
||||||
|
return date
|
||||||
|
},
|
||||||
|
"strings_First": func(s string, n int) string {
|
||||||
|
if n >= len(s) {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return s[:n]
|
||||||
|
},
|
||||||
|
"strings_Last": func(s string, n int) string {
|
||||||
|
if n >= len(s) {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return s[len(s)-n:]
|
||||||
|
},
|
||||||
|
"strings_Truncate": func(s string, n int) string {
|
||||||
|
if n >= len(s) {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return s[:n] + "..."
|
||||||
|
},
|
||||||
|
"strings_TrimPrefix": strings.TrimPrefix,
|
||||||
|
"strings_TrimSuffix": strings.TrimSuffix,
|
||||||
|
|
||||||
|
// Dict
|
||||||
|
"dict_Str": func(values ...any) 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)
|
||||||
|
dict[key] = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return dict
|
||||||
|
},
|
||||||
|
|
||||||
|
// Body
|
||||||
|
"body_FormData": func(kv map[string]string) string {
|
||||||
|
var data bytes.Buffer
|
||||||
|
writer := multipart.NewWriter(&data)
|
||||||
|
|
||||||
|
for k, v := range kv {
|
||||||
|
_ = writer.WriteField(k, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
_ = writer.Close()
|
||||||
|
g.bodyDataHeader = writer.FormDataContentType()
|
||||||
|
|
||||||
|
return data.String()
|
||||||
|
},
|
||||||
|
|
||||||
|
// FakeIt / Product
|
||||||
|
"fakeit_ProductName": g.localFaker.ProductName,
|
||||||
|
"fakeit_ProductDescription": g.localFaker.ProductDescription,
|
||||||
|
"fakeit_ProductCategory": g.localFaker.ProductCategory,
|
||||||
|
"fakeit_ProductFeature": g.localFaker.ProductFeature,
|
||||||
|
"fakeit_ProductMaterial": g.localFaker.ProductMaterial,
|
||||||
|
"fakeit_ProductUPC": g.localFaker.ProductUPC,
|
||||||
|
"fakeit_ProductAudience": g.localFaker.ProductAudience,
|
||||||
|
"fakeit_ProductDimension": g.localFaker.ProductDimension,
|
||||||
|
"fakeit_ProductUseCase": g.localFaker.ProductUseCase,
|
||||||
|
"fakeit_ProductBenefit": g.localFaker.ProductBenefit,
|
||||||
|
"fakeit_ProductSuffix": g.localFaker.ProductSuffix,
|
||||||
|
|
||||||
|
// FakeIt / Person
|
||||||
|
"fakeit_Name": g.localFaker.Name,
|
||||||
|
"fakeit_NamePrefix": g.localFaker.NamePrefix,
|
||||||
|
"fakeit_NameSuffix": g.localFaker.NameSuffix,
|
||||||
|
"fakeit_FirstName": g.localFaker.FirstName,
|
||||||
|
"fakeit_MiddleName": g.localFaker.MiddleName,
|
||||||
|
"fakeit_LastName": g.localFaker.LastName,
|
||||||
|
"fakeit_Gender": g.localFaker.Gender,
|
||||||
|
"fakeit_SSN": g.localFaker.SSN,
|
||||||
|
"fakeit_Hobby": g.localFaker.Hobby,
|
||||||
|
"fakeit_Email": g.localFaker.Email,
|
||||||
|
"fakeit_Phone": g.localFaker.Phone,
|
||||||
|
"fakeit_PhoneFormatted": g.localFaker.PhoneFormatted,
|
||||||
|
|
||||||
|
// FakeIt / Auth
|
||||||
|
"fakeit_Username": g.localFaker.Username,
|
||||||
|
"fakeit_Password": g.localFaker.Password,
|
||||||
|
|
||||||
|
// FakeIt / Address
|
||||||
|
"fakeit_City": g.localFaker.City,
|
||||||
|
"fakeit_Country": g.localFaker.Country,
|
||||||
|
"fakeit_CountryAbr": g.localFaker.CountryAbr,
|
||||||
|
"fakeit_State": g.localFaker.State,
|
||||||
|
"fakeit_StateAbr": g.localFaker.StateAbr,
|
||||||
|
"fakeit_Street": g.localFaker.Street,
|
||||||
|
"fakeit_StreetName": g.localFaker.StreetName,
|
||||||
|
"fakeit_StreetNumber": g.localFaker.StreetNumber,
|
||||||
|
"fakeit_StreetPrefix": g.localFaker.StreetPrefix,
|
||||||
|
"fakeit_StreetSuffix": g.localFaker.StreetSuffix,
|
||||||
|
"fakeit_Zip": g.localFaker.Zip,
|
||||||
|
"fakeit_Latitude": g.localFaker.Latitude,
|
||||||
|
"fakeit_LatitudeInRange": func(min, max float64) float64 {
|
||||||
|
value, err := g.localFaker.LatitudeInRange(min, max)
|
||||||
|
if err != nil {
|
||||||
|
var zero float64
|
||||||
|
return zero
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
},
|
||||||
|
"fakeit_Longitude": g.localFaker.Longitude,
|
||||||
|
"fakeit_LongitudeInRange": func(min, max float64) float64 {
|
||||||
|
value, err := g.localFaker.LongitudeInRange(min, max)
|
||||||
|
if err != nil {
|
||||||
|
var zero float64
|
||||||
|
return zero
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
},
|
||||||
|
|
||||||
|
// FakeIt / Game
|
||||||
|
"fakeit_Gamertag": g.localFaker.Gamertag,
|
||||||
|
|
||||||
|
// FakeIt / Beer
|
||||||
|
"fakeit_BeerAlcohol": g.localFaker.BeerAlcohol,
|
||||||
|
"fakeit_BeerBlg": g.localFaker.BeerBlg,
|
||||||
|
"fakeit_BeerHop": g.localFaker.BeerHop,
|
||||||
|
"fakeit_BeerIbu": g.localFaker.BeerIbu,
|
||||||
|
"fakeit_BeerMalt": g.localFaker.BeerMalt,
|
||||||
|
"fakeit_BeerName": g.localFaker.BeerName,
|
||||||
|
"fakeit_BeerStyle": g.localFaker.BeerStyle,
|
||||||
|
"fakeit_BeerYeast": g.localFaker.BeerYeast,
|
||||||
|
|
||||||
|
// FakeIt / Car
|
||||||
|
"fakeit_CarMaker": g.localFaker.CarMaker,
|
||||||
|
"fakeit_CarModel": g.localFaker.CarModel,
|
||||||
|
"fakeit_CarType": g.localFaker.CarType,
|
||||||
|
"fakeit_CarFuelType": g.localFaker.CarFuelType,
|
||||||
|
"fakeit_CarTransmissionType": g.localFaker.CarTransmissionType,
|
||||||
|
|
||||||
|
// FakeIt / Words
|
||||||
|
"fakeit_Noun": g.localFaker.Noun,
|
||||||
|
"fakeit_NounCommon": g.localFaker.NounCommon,
|
||||||
|
"fakeit_NounConcrete": g.localFaker.NounConcrete,
|
||||||
|
"fakeit_NounAbstract": g.localFaker.NounAbstract,
|
||||||
|
"fakeit_NounCollectivePeople": g.localFaker.NounCollectivePeople,
|
||||||
|
"fakeit_NounCollectiveAnimal": g.localFaker.NounCollectiveAnimal,
|
||||||
|
"fakeit_NounCollectiveThing": g.localFaker.NounCollectiveThing,
|
||||||
|
"fakeit_NounCountable": g.localFaker.NounCountable,
|
||||||
|
"fakeit_NounUncountable": g.localFaker.NounUncountable,
|
||||||
|
"fakeit_Verb": g.localFaker.Verb,
|
||||||
|
"fakeit_VerbAction": g.localFaker.VerbAction,
|
||||||
|
"fakeit_VerbLinking": g.localFaker.VerbLinking,
|
||||||
|
"fakeit_VerbHelping": g.localFaker.VerbHelping,
|
||||||
|
"fakeit_Adverb": g.localFaker.Adverb,
|
||||||
|
"fakeit_AdverbManner": g.localFaker.AdverbManner,
|
||||||
|
"fakeit_AdverbDegree": g.localFaker.AdverbDegree,
|
||||||
|
"fakeit_AdverbPlace": g.localFaker.AdverbPlace,
|
||||||
|
"fakeit_AdverbTimeDefinite": g.localFaker.AdverbTimeDefinite,
|
||||||
|
"fakeit_AdverbTimeIndefinite": g.localFaker.AdverbTimeIndefinite,
|
||||||
|
"fakeit_AdverbFrequencyDefinite": g.localFaker.AdverbFrequencyDefinite,
|
||||||
|
"fakeit_AdverbFrequencyIndefinite": g.localFaker.AdverbFrequencyIndefinite,
|
||||||
|
"fakeit_Preposition": g.localFaker.Preposition,
|
||||||
|
"fakeit_PrepositionSimple": g.localFaker.PrepositionSimple,
|
||||||
|
"fakeit_PrepositionDouble": g.localFaker.PrepositionDouble,
|
||||||
|
"fakeit_PrepositionCompound": g.localFaker.PrepositionCompound,
|
||||||
|
"fakeit_Adjective": g.localFaker.Adjective,
|
||||||
|
"fakeit_AdjectiveDescriptive": g.localFaker.AdjectiveDescriptive,
|
||||||
|
"fakeit_AdjectiveQuantitative": g.localFaker.AdjectiveQuantitative,
|
||||||
|
"fakeit_AdjectiveProper": g.localFaker.AdjectiveProper,
|
||||||
|
"fakeit_AdjectiveDemonstrative": g.localFaker.AdjectiveDemonstrative,
|
||||||
|
"fakeit_AdjectivePossessive": g.localFaker.AdjectivePossessive,
|
||||||
|
"fakeit_AdjectiveInterrogative": g.localFaker.AdjectiveInterrogative,
|
||||||
|
"fakeit_AdjectiveIndefinite": g.localFaker.AdjectiveIndefinite,
|
||||||
|
"fakeit_Pronoun": g.localFaker.Pronoun,
|
||||||
|
"fakeit_PronounPersonal": g.localFaker.PronounPersonal,
|
||||||
|
"fakeit_PronounObject": g.localFaker.PronounObject,
|
||||||
|
"fakeit_PronounPossessive": g.localFaker.PronounPossessive,
|
||||||
|
"fakeit_PronounReflective": g.localFaker.PronounReflective,
|
||||||
|
"fakeit_PronounDemonstrative": g.localFaker.PronounDemonstrative,
|
||||||
|
"fakeit_PronounInterrogative": g.localFaker.PronounInterrogative,
|
||||||
|
"fakeit_PronounRelative": g.localFaker.PronounRelative,
|
||||||
|
"fakeit_Connective": g.localFaker.Connective,
|
||||||
|
"fakeit_ConnectiveTime": g.localFaker.ConnectiveTime,
|
||||||
|
"fakeit_ConnectiveComparative": g.localFaker.ConnectiveComparative,
|
||||||
|
"fakeit_ConnectiveComplaint": g.localFaker.ConnectiveComplaint,
|
||||||
|
"fakeit_ConnectiveListing": g.localFaker.ConnectiveListing,
|
||||||
|
"fakeit_ConnectiveCasual": g.localFaker.ConnectiveCasual,
|
||||||
|
"fakeit_ConnectiveExamplify": g.localFaker.ConnectiveExamplify,
|
||||||
|
"fakeit_Word": g.localFaker.Word,
|
||||||
|
"fakeit_Sentence": g.localFaker.Sentence,
|
||||||
|
"fakeit_Paragraph": g.localFaker.Paragraph,
|
||||||
|
"fakeit_LoremIpsumWord": g.localFaker.LoremIpsumWord,
|
||||||
|
"fakeit_LoremIpsumSentence": g.localFaker.LoremIpsumSentence,
|
||||||
|
"fakeit_LoremIpsumParagraph": g.localFaker.LoremIpsumParagraph,
|
||||||
|
"fakeit_Question": g.localFaker.Question,
|
||||||
|
"fakeit_Quote": g.localFaker.Quote,
|
||||||
|
"fakeit_Phrase": g.localFaker.Phrase,
|
||||||
|
|
||||||
|
// FakeIt / Foods
|
||||||
|
"fakeit_Fruit": g.localFaker.Fruit,
|
||||||
|
"fakeit_Vegetable": g.localFaker.Vegetable,
|
||||||
|
"fakeit_Breakfast": g.localFaker.Breakfast,
|
||||||
|
"fakeit_Lunch": g.localFaker.Lunch,
|
||||||
|
"fakeit_Dinner": g.localFaker.Dinner,
|
||||||
|
"fakeit_Snack": g.localFaker.Snack,
|
||||||
|
"fakeit_Dessert": g.localFaker.Dessert,
|
||||||
|
|
||||||
|
// FakeIt / Misc
|
||||||
|
"fakeit_Bool": g.localFaker.Bool,
|
||||||
|
"fakeit_UUID": g.localFaker.UUID,
|
||||||
|
"fakeit_FlipACoin": g.localFaker.FlipACoin,
|
||||||
|
|
||||||
|
// FakeIt / Colors
|
||||||
|
"fakeit_Color": g.localFaker.Color,
|
||||||
|
"fakeit_HexColor": g.localFaker.HexColor,
|
||||||
|
"fakeit_RGBColor": g.localFaker.RGBColor,
|
||||||
|
"fakeit_SafeColor": g.localFaker.SafeColor,
|
||||||
|
"fakeit_NiceColors": g.localFaker.NiceColors,
|
||||||
|
|
||||||
|
// FakeIt / Internet
|
||||||
|
"fakeit_URL": g.localFaker.URL,
|
||||||
|
"fakeit_DomainName": g.localFaker.DomainName,
|
||||||
|
"fakeit_DomainSuffix": g.localFaker.DomainSuffix,
|
||||||
|
"fakeit_IPv4Address": g.localFaker.IPv4Address,
|
||||||
|
"fakeit_IPv6Address": g.localFaker.IPv6Address,
|
||||||
|
"fakeit_MacAddress": g.localFaker.MacAddress,
|
||||||
|
"fakeit_HTTPStatusCode": g.localFaker.HTTPStatusCode,
|
||||||
|
"fakeit_HTTPStatusCodeSimple": g.localFaker.HTTPStatusCodeSimple,
|
||||||
|
"fakeit_LogLevel": g.localFaker.LogLevel,
|
||||||
|
"fakeit_HTTPMethod": g.localFaker.HTTPMethod,
|
||||||
|
"fakeit_HTTPVersion": g.localFaker.HTTPVersion,
|
||||||
|
"fakeit_UserAgent": g.localFaker.UserAgent,
|
||||||
|
"fakeit_ChromeUserAgent": g.localFaker.ChromeUserAgent,
|
||||||
|
"fakeit_FirefoxUserAgent": g.localFaker.FirefoxUserAgent,
|
||||||
|
"fakeit_OperaUserAgent": g.localFaker.OperaUserAgent,
|
||||||
|
"fakeit_SafariUserAgent": g.localFaker.SafariUserAgent,
|
||||||
|
|
||||||
|
// FakeIt / HTML
|
||||||
|
"fakeit_InputName": g.localFaker.InputName,
|
||||||
|
|
||||||
|
// FakeIt / Date/Time
|
||||||
|
"fakeit_Date": g.localFaker.Date,
|
||||||
|
"fakeit_PastDate": g.localFaker.PastDate,
|
||||||
|
"fakeit_FutureDate": g.localFaker.FutureDate,
|
||||||
|
"fakeit_DateRange": g.localFaker.DateRange,
|
||||||
|
"fakeit_NanoSecond": g.localFaker.NanoSecond,
|
||||||
|
"fakeit_Second": g.localFaker.Second,
|
||||||
|
"fakeit_Minute": g.localFaker.Minute,
|
||||||
|
"fakeit_Hour": g.localFaker.Hour,
|
||||||
|
"fakeit_Month": g.localFaker.Month,
|
||||||
|
"fakeit_MonthString": g.localFaker.MonthString,
|
||||||
|
"fakeit_Day": g.localFaker.Day,
|
||||||
|
"fakeit_WeekDay": g.localFaker.WeekDay,
|
||||||
|
"fakeit_Year": g.localFaker.Year,
|
||||||
|
"fakeit_TimeZone": g.localFaker.TimeZone,
|
||||||
|
"fakeit_TimeZoneAbv": g.localFaker.TimeZoneAbv,
|
||||||
|
"fakeit_TimeZoneFull": g.localFaker.TimeZoneFull,
|
||||||
|
"fakeit_TimeZoneOffset": g.localFaker.TimeZoneOffset,
|
||||||
|
"fakeit_TimeZoneRegion": g.localFaker.TimeZoneRegion,
|
||||||
|
|
||||||
|
// FakeIt / Payment
|
||||||
|
"fakeit_Price": g.localFaker.Price,
|
||||||
|
"fakeit_CreditCardCvv": g.localFaker.CreditCardCvv,
|
||||||
|
"fakeit_CreditCardExp": g.localFaker.CreditCardExp,
|
||||||
|
"fakeit_CreditCardNumber": g.localFaker.CreditCardNumber,
|
||||||
|
"fakeit_CreditCardType": g.localFaker.CreditCardType,
|
||||||
|
"fakeit_CurrencyLong": g.localFaker.CurrencyLong,
|
||||||
|
"fakeit_CurrencyShort": g.localFaker.CurrencyShort,
|
||||||
|
"fakeit_AchRouting": g.localFaker.AchRouting,
|
||||||
|
"fakeit_AchAccount": g.localFaker.AchAccount,
|
||||||
|
"fakeit_BitcoinAddress": g.localFaker.BitcoinAddress,
|
||||||
|
"fakeit_BitcoinPrivateKey": g.localFaker.BitcoinPrivateKey,
|
||||||
|
|
||||||
|
// FakeIt / Finance
|
||||||
|
"fakeit_Cusip": g.localFaker.Cusip,
|
||||||
|
"fakeit_Isin": g.localFaker.Isin,
|
||||||
|
|
||||||
|
// FakeIt / Company
|
||||||
|
"fakeit_BS": g.localFaker.BS,
|
||||||
|
"fakeit_Blurb": g.localFaker.Blurb,
|
||||||
|
"fakeit_BuzzWord": g.localFaker.BuzzWord,
|
||||||
|
"fakeit_Company": g.localFaker.Company,
|
||||||
|
"fakeit_CompanySuffix": g.localFaker.CompanySuffix,
|
||||||
|
"fakeit_JobDescriptor": g.localFaker.JobDescriptor,
|
||||||
|
"fakeit_JobLevel": g.localFaker.JobLevel,
|
||||||
|
"fakeit_JobTitle": g.localFaker.JobTitle,
|
||||||
|
"fakeit_Slogan": g.localFaker.Slogan,
|
||||||
|
|
||||||
|
// FakeIt / Hacker
|
||||||
|
"fakeit_HackerAbbreviation": g.localFaker.HackerAbbreviation,
|
||||||
|
"fakeit_HackerAdjective": g.localFaker.HackerAdjective,
|
||||||
|
"fakeit_HackerNoun": g.localFaker.HackerNoun,
|
||||||
|
"fakeit_HackerPhrase": g.localFaker.HackerPhrase,
|
||||||
|
"fakeit_HackerVerb": g.localFaker.HackerVerb,
|
||||||
|
|
||||||
|
// FakeIt / Hipster
|
||||||
|
"fakeit_HipsterWord": g.localFaker.HipsterWord,
|
||||||
|
"fakeit_HipsterSentence": g.localFaker.HipsterSentence,
|
||||||
|
"fakeit_HipsterParagraph": g.localFaker.HipsterParagraph,
|
||||||
|
|
||||||
|
// FakeIt / App
|
||||||
|
"fakeit_AppName": g.localFaker.AppName,
|
||||||
|
"fakeit_AppVersion": g.localFaker.AppVersion,
|
||||||
|
"fakeit_AppAuthor": g.localFaker.AppAuthor,
|
||||||
|
|
||||||
|
// FakeIt / Animal
|
||||||
|
"fakeit_PetName": g.localFaker.PetName,
|
||||||
|
"fakeit_Animal": g.localFaker.Animal,
|
||||||
|
"fakeit_AnimalType": g.localFaker.AnimalType,
|
||||||
|
"fakeit_FarmAnimal": g.localFaker.FarmAnimal,
|
||||||
|
"fakeit_Cat": g.localFaker.Cat,
|
||||||
|
"fakeit_Dog": g.localFaker.Dog,
|
||||||
|
"fakeit_Bird": g.localFaker.Bird,
|
||||||
|
|
||||||
|
// FakeIt / Emoji
|
||||||
|
"fakeit_Emoji": g.localFaker.Emoji,
|
||||||
|
"fakeit_EmojiDescription": g.localFaker.EmojiDescription,
|
||||||
|
"fakeit_EmojiCategory": g.localFaker.EmojiCategory,
|
||||||
|
"fakeit_EmojiAlias": g.localFaker.EmojiAlias,
|
||||||
|
"fakeit_EmojiTag": g.localFaker.EmojiTag,
|
||||||
|
|
||||||
|
// FakeIt / Language
|
||||||
|
"fakeit_Language": g.localFaker.Language,
|
||||||
|
"fakeit_LanguageAbbreviation": g.localFaker.LanguageAbbreviation,
|
||||||
|
"fakeit_ProgrammingLanguage": g.localFaker.ProgrammingLanguage,
|
||||||
|
|
||||||
|
// FakeIt / Number
|
||||||
|
"fakeit_Number": g.localFaker.Number,
|
||||||
|
"fakeit_Int": g.localFaker.Int,
|
||||||
|
"fakeit_IntN": g.localFaker.IntN,
|
||||||
|
"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_Uint8": g.localFaker.Uint8,
|
||||||
|
"fakeit_Uint16": g.localFaker.Uint16,
|
||||||
|
"fakeit_Uint32": g.localFaker.Uint32,
|
||||||
|
"fakeit_Uint64": g.localFaker.Uint64,
|
||||||
|
"fakeit_Float32": g.localFaker.Float32,
|
||||||
|
"fakeit_Float32Range": g.localFaker.Float32Range,
|
||||||
|
"fakeit_Float64": g.localFaker.Float64,
|
||||||
|
"fakeit_Float64Range": g.localFaker.Float64Range,
|
||||||
|
"fakeit_HexUint": g.localFaker.HexUint,
|
||||||
|
|
||||||
|
// FakeIt / String
|
||||||
|
"fakeit_Digit": g.localFaker.Digit,
|
||||||
|
"fakeit_DigitN": g.localFaker.DigitN,
|
||||||
|
"fakeit_Letter": g.localFaker.Letter,
|
||||||
|
"fakeit_LetterN": g.localFaker.LetterN,
|
||||||
|
"fakeit_Lexify": g.localFaker.Lexify,
|
||||||
|
"fakeit_Numerify": g.localFaker.Numerify,
|
||||||
|
|
||||||
|
// FakeIt / Celebrity
|
||||||
|
"fakeit_CelebrityActor": g.localFaker.CelebrityActor,
|
||||||
|
"fakeit_CelebrityBusiness": g.localFaker.CelebrityBusiness,
|
||||||
|
"fakeit_CelebritySport": g.localFaker.CelebritySport,
|
||||||
|
|
||||||
|
// FakeIt / Minecraft
|
||||||
|
"fakeit_MinecraftOre": g.localFaker.MinecraftOre,
|
||||||
|
"fakeit_MinecraftWood": g.localFaker.MinecraftWood,
|
||||||
|
"fakeit_MinecraftArmorTier": g.localFaker.MinecraftArmorTier,
|
||||||
|
"fakeit_MinecraftArmorPart": g.localFaker.MinecraftArmorPart,
|
||||||
|
"fakeit_MinecraftWeapon": g.localFaker.MinecraftWeapon,
|
||||||
|
"fakeit_MinecraftTool": g.localFaker.MinecraftTool,
|
||||||
|
"fakeit_MinecraftDye": g.localFaker.MinecraftDye,
|
||||||
|
"fakeit_MinecraftFood": g.localFaker.MinecraftFood,
|
||||||
|
"fakeit_MinecraftAnimal": g.localFaker.MinecraftAnimal,
|
||||||
|
"fakeit_MinecraftVillagerJob": g.localFaker.MinecraftVillagerJob,
|
||||||
|
"fakeit_MinecraftVillagerStation": g.localFaker.MinecraftVillagerStation,
|
||||||
|
"fakeit_MinecraftVillagerLevel": g.localFaker.MinecraftVillagerLevel,
|
||||||
|
"fakeit_MinecraftMobPassive": g.localFaker.MinecraftMobPassive,
|
||||||
|
"fakeit_MinecraftMobNeutral": g.localFaker.MinecraftMobNeutral,
|
||||||
|
"fakeit_MinecraftMobHostile": g.localFaker.MinecraftMobHostile,
|
||||||
|
"fakeit_MinecraftMobBoss": g.localFaker.MinecraftMobBoss,
|
||||||
|
"fakeit_MinecraftBiome": g.localFaker.MinecraftBiome,
|
||||||
|
"fakeit_MinecraftWeather": g.localFaker.MinecraftWeather,
|
||||||
|
|
||||||
|
// FakeIt / Book
|
||||||
|
"fakeit_BookTitle": g.localFaker.BookTitle,
|
||||||
|
"fakeit_BookAuthor": g.localFaker.BookAuthor,
|
||||||
|
"fakeit_BookGenre": g.localFaker.BookGenre,
|
||||||
|
|
||||||
|
// FakeIt / Movie
|
||||||
|
"fakeit_MovieName": g.localFaker.MovieName,
|
||||||
|
"fakeit_MovieGenre": g.localFaker.MovieGenre,
|
||||||
|
|
||||||
|
// FakeIt / Error
|
||||||
|
"fakeit_Error": g.localFaker.Error,
|
||||||
|
"fakeit_ErrorDatabase": g.localFaker.ErrorDatabase,
|
||||||
|
"fakeit_ErrorGRPC": g.localFaker.ErrorGRPC,
|
||||||
|
"fakeit_ErrorHTTP": g.localFaker.ErrorHTTP,
|
||||||
|
"fakeit_ErrorHTTPClient": g.localFaker.ErrorHTTPClient,
|
||||||
|
"fakeit_ErrorHTTPServer": g.localFaker.ErrorHTTPServer,
|
||||||
|
"fakeit_ErrorRuntime": g.localFaker.ErrorRuntime,
|
||||||
|
|
||||||
|
// FakeIt / School
|
||||||
|
"fakeit_School": g.localFaker.School,
|
||||||
|
|
||||||
|
// FakeIt / Song
|
||||||
|
"fakeit_SongName": g.localFaker.SongName,
|
||||||
|
"fakeit_SongArtist": g.localFaker.SongArtist,
|
||||||
|
"fakeit_SongGenre": g.localFaker.SongGenre,
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user