Do is a Go program that parses custom .do
files and executes HTTP requests based on the parsed content.
I created do
because I needed a way to version HTTP requests and distribute them across teams in a simple, plaintext format. .do
files provide a human-readable and easily shareable way to define HTTP requests, making it convenient for collaboration and version control.
Execute:
go install github.com/jibaru/do/cmd/do@latest
Also you can rename to dohttp.
Unix:
mv ~/go/bin/do ~/go/bin/dohttp
Windows (CMD):
move %USERPROFILE%\go\bin\do.exe %USERPROFILE%\go\bin\dohttp.exe
Execute your filename.do
files:
do -f path/to/do/file
You can include env variables to use in your .do
files:
do -f path/to/do/file -e path/to/env/file
let {
base = "https://api.example.com";
userId = 23;
token = env("API_TOKEN, "default-http-token");
page = 1;
deleted = false;
}
do {
method = "POST";
url = "$base/users/:id";
params = {"id": userId};
query = {
"p": page,
"deleted": deleted
};
headers = {
"Authorization": "Bearer $token",
"Content-Type": "application/json"
};
body = `{
"name": "John Doe",
"email": "[email protected]"
}`;
}
The .do file format consists of two main sections: let and do.
The let section is used to declare variables that can be referenced in the do section. Variables are declared in the following format:
let {
base = "https://api.example.com";
userId = 23;
token = env("API_TOKEN, "default-http-token");
page = 1;
deleted = false;
// Add more variables here...
}
You should use the variable = value;
format to define a variable with its value. Values has specific types that are defined below:
Type | Description | Example |
---|---|---|
int | Integer value | 12 |
float | Decimal value | 92.3 |
bool | Boolean value (true or false) | true |
string | A character sequence. You can use ` to wrap a string that contains " | "example" |
There are another values that only should be accepted in do section:
Type | Description | Example |
---|---|---|
map | A collection of key-value pairs | {"key1": 12, "key2": false, "key3": "hello"} |
Function | Description | Example |
---|---|---|
env | Get an environment variable. If the variable is not found, it returns a default value | env("MY_VAR", "default") |
file | Get a file path. It is used for multipart requests. | file("path/to/file.txt") |
date | Generate a new string with the specified date format. | date("ISO8601") |
uuid | Generate a new uuid v4 string. | uuid() |
The do section specifies the HTTP request to be executed. It contains various fields such as method, URL, params, query, headers, and body. Here's an example of how to define the do section:
do {
method = "POST";
url = "$base/users/:id";
params = {"id": userId};
query = {
"p": page,
"deleted": deleted
};
headers = {
"Authorization": "Bearer $token",
"Content-Type": "application/json"
};
body = `{
"name": "John Doe",
"email": "[email protected]"
}`;
}
In the do section, variables defined in the let section can be referenced using their names.
You can insert variables in strings using the syntax "$variableName"
.
Take in note that body
could be a string that wraps a json using backticks or could be a map for multipart requests.
You should use the variable = value;
format to define a variable with its value. Values has specific types that are defined below:
Field | Type | Description | Required | Example |
---|---|---|---|---|
method | string | The http request method | Yes | "POST" |
url | string | The url to request. It should accept replace params that should be used with : + name |
Yes | "https://example.com" |
params | map | The params to replace in the url (without : ). |
No | {"id": 26} |
query | map | The query params for the request. | No | {"active": false, "order": "asc"} |
headers | map | The headers for the request. | No | {"Authorization": "application/json"} |
body | string or map | The body for the request. If it is type map, a multipart-form should be used. | No | `{"name": "john"}` |
If you want to send a multipart request, you should use a map as the body. The map should contain the field name as the key and the file path as the value using the file function. Here's an example:
let {
myFile = file("path/to/file.txt");
}
do {
method = "POST";
url = "https://example.com/upload";
body = {
"file": myFile
};
}
The output of the do
command will be the request + response in a json format.
{
"do_file": {
"let": {
"variables": {
"token": "token-value"
}
},
"do": {
"method": "POST",
"url": "https://www.fakepage.com/keys/:id",
"params": {
"id": 1
},
"query": {
"limit": 1
},
"headers": {
"Authorization": "Bearer token-value"
},
"body": "value"
}
},
"response": {
"status_code": 200,
"body": "{\"key\": 123}",
"headers": {
"Content-Type": "application/json; charset=utf-8"
}
},
"error": null
}
The do_file
shows the parsed request from the .do file.
The response
shows the response from the request if everything works well.
The error
shows the error if parsing the .do file or executing the request fails. It is only a string.
If you want to use the response into another program, make sure validate error is null before trying to parse the response and request.
-f
or-file
: The file path to execute.-v
or-version
: Show the version of the program.-h
or-help
: Show the help message.-e
or-env
: Set the environment variables using a file path that contains the variables.
You can add support for .do
files using the following extension: