README ¶
apitest
The apitest tool helps you to build automated apitests that can be run after every build to ensure a constant product quality.
A single testcase is also a perfect definition of an occuring problem and helps the developers to fix your issues faster!
Configuration file
For configuring the apitest tool, add the following section to your apitest.yml
configuration file.
The report parameters of this config can be overwritten via a command line flag. So you should set your intended standar
apitest:
server: "http://5.simon.pf-berlin.de/api/v1" # The base url to the api you want to fire the apitests against. Important: don’t add a trailing ‘/’
log:
short: true # Configures minimal logs by default for all tests
report: # Configures the maschine report. For usage with jenkis or any other CI tool
file: "apitest_report.xml" # Filename of the report file. The file gets saved in the same directory of the apitest binary
format: "json.junit" # Format of the report. (Supported formats: json, junit or stats)
store: # initial values for the datastore, parsed as map[string]interface{}
email.server: smtp.google.com
oauth2_client: # Map of client-config for oAuth clients
my_client: # oauth Client ID
endpoint: # endpoints on the oauth server
auth_url: "http://auth.myserver.de/oauth/auth"
token_url: "http://auth.myserver.de/oauth/token"
secret: "foobar" # oauth Client secret
redirect_url: "http://myfancyapp.de/auth/receive-fancy-token" # redirect, usually on client side
The YAML config is optional. All config values can be overwritten/set by command line parameters: see Overwrite config parameters
Command line interface
You start the apitest tool with the following command
./apitest
This starts the command with the following default settings:
- Runs all tests that are in the current directory, or in any of its subdirectories
- Logs to console
- Writes the machine log, to the given file in the
apitest.yml
- Logs only the request & responses if a test fails
Configure which tests should be run
Parameter | Description | |
---|---|---|
--directory testDirectory |
-d testDirectory |
Defines which directory should be used for running the tests in it. The tool walks recursively trough all subdirectories and runs alls tests that have a manifest.json file in alphabetical order of the folder names. (Depth-First-Search) |
--single manifest.json |
-s manifest.json |
Run only a single test. The path needs to point directly to the manifest file. (Not the directory containing it) |
Stop on fail
Parameter | Description |
---|---|
stop-on-fail |
Stop execution of later test suites if a test suite fails |
Keep running
keep-running
: Wait for a keyboard interrupt after each test suite invocation. This can be useful for keeping the HTTP / SMTP server for manual inspection.
Configure logging
Per default request and response of a request will be logged on test failure. If you want to see more information you can configure the tool with additional log flags
Parameter | Description | |
---|---|---|
--log-network |
-n |
Log all network traffic |
--log-datastore |
Logs datastore operations into datastore | |
--log-verbose |
-v |
--log-network , --log-datastore and a few additional trace informations |
--log-short |
Show minimal logs, useful for CI chains | |
--log-timestamp |
-t |
Log the timestamp of the log message into the console |
--curl-bash |
Log the request as curl command | |
--limit-request |
Limit the lines of request log output. Configure limit in apitest.yml |
|
--limit-response |
Limit the lines of response log output. Configure limit in apitest.yml |
You can also set the log verbosity per single testcase. The greater verbosity wins.
Overwrite config parameters
Parameter | Description | |
---|---|---|
--config newConfigFile |
-c newConfigFile |
Overwrites the path of the config file (default ./apitest.yml ) with newConfigFile |
--server URL |
Overwrites base url to the api | |
--report-file newReportFile |
Overwrites the report file name from the apitest.yml config with newReportFile |
|
--report-format junit |
Overwrites the report format from the apitest.yml config with junit |
|
--replace-host host |
Overwrites built-in server host in template function replace_host |
Additional parameters
Parameter | Description |
---|---|
--report-format-stats-group 3 |
Sets the number of groups for manifests distrubution when using report format stats |
Examples
- Run all tests in the directory apitests display all server communication and save the maschine report as junit for later parsing it with jenkins
./apitest --directory apitests --verbosity 2 --report-format junit
- Only run a single test apitests/test1/manifest.json with no console output and save the maschine report to the standard file defined in the
apitest.yml
./apitest --single apitests/test1/manifest.json --log-console-enable false
- Run all tests in the directory apitests with http server host replacement for those templates using replace_host template function
./apitest -d apitests --replace-host my.fancy.host
Manifest
Manifest is loaded as template, so you can use variables, Go range and if and others.
{
// General info about the testuite. Try to explain your problem indepth here. So that someone who works on the test years from now knows what is happening
"description": "search api tests for filename",
// Testname. Should be the ticket number if the test is based on a ticket
"name": "ticket_48565",
// init store
"store": {
"custom": "data"
},
// Testsuites your want to run upfront (e.g. a setup). Paths are relative to the current test manifest
"require": [
"setup_manifests/purge.yaml",
"setup_manifests/config.yaml",
"setup_manifests/upload_datamodel.yaml"
],
// Array of single testcases. Add es much as you want. They get executed in chronological order
"tests": [
// [SINGLE TESTCASE]: See below for more information
// [SINGLE TESTCASE]: See below for more information
// [SINGLE TESTCASE]: See below for more information
// We also support the external loading of a complete test:
"@pathToTest.json",
// By prefixing it with a number, the testtool runs that many instances of
// the included test file in parallel to each other.
//
// Only tests directly included by the manifest are allowed to run in parallel.
"5@pathToTestsThatShouldRunInParallel.json"
]
}
Testcase Definition
manifest.json
{
// Define if the test suite should continue even if this test fails. (default: false)
"continue_on_failure": true,
// Name to identify this single test. Is important for the log. Try to give an explaning name
"name": "Testname",
// Store custom values to the datastore
"store": {
"key1": "value1",
"key2": "value2"
},
// Optional temporary HTTP Server (see below)
"http_server": {
"addr": ":1234",
"dir": ".",
"testmode": false
},
// Optional temporary SMTP Server (see below)
"smtp_server": {
"addr": ":9025",
"max_message_size": 1000000,
},
// Specify a unique log behavior only for this single test.
"log_network": true,
"log_verbose": false,
// Show or disable minimal logs for this test
"log_short": false,
// Defines what gets send to the server
"request": {
// What endpoint we want to target. You find all possible endpoints in the api documentation
"endpoint": "suggest",
// the server url to connect can be set directly for a request, overwriting the configured server url
"server_url": "",
// How the endpoint should be accessed. The api documentations tells your which methods are possible for an endpoint. All HTTP methods are possible.
"method": "GET",
// If set to true, don't follow redirects.
"no_redirect": false,
// Parameters that will be added to the url. e.g. http:// 5.testing.pf-berlin.de/api/v1/session?token=testtoken&number=2 would be defined as follows
"query_params": {
"number": 2,
"token": "testtoken"
},
// With query_params_from_store set a query parameter to the value of the datastore field
"query_params_from_store": {
"format": "formatFromDatastore",
// If the datastore key starts with an ?, wo do not throw an error if the key could not be found, but just
// do not set the query param. If the key "a" is not found it datastore, the query parameter test will not be set
"test": "?a"
},
// Additional headers that should be added to the request
"header": {
"header1": "value",
"header2": ["value1", "value2"]
},
// Cookies can be added to the request
"cookies": {
// name of a cookie to be set
"cookie1": {
// A cookie can be get parsed from store if it was saved before
// It will ignore the cookie if it is not set
"value_from_store": "sess_cookie",
// Or its values can be directly set, overriding the one from store, if defined
"value": "value"
},
"cookie2": {
"value_from_store": "ads_cookie",
}
},
// Special headers `X-Test-Set-Cookie` can be populated in the request (on per entry)
// It is used in the builting `http_server` to automatically set those cookies on response
// So it is useful for mocking them for further testing
"header-x-test-set-cookie": [
{
"name": "sess",
"value": "myauthtoken"
},
{
"name": "jwtoken",
"value": "tokenized",
"path": "/auth",
"domain": "mydomain",
"expires": "2021-11-10T10:00:00Z",
"max_age": 86400,
"secure": false,
"http_only": true,
"same_site": 1
}
],
// With header_from_store you set a header to the value of the datastore field
// In this example we set the "Content-Type" header to the value "application/json"
// As "application/json" is stored as string in the datastore on index "contentType"
"header_from_store": {
"Content-Type": "contentType",
// If the datastore key starts with an ?, wo do not throw an error if the key could not be found, but just
// do not set the header. If the key "a" is not found it datastore, the header Range will not be set
"Range": "?a"
},
// All the content you want to send in the http body. Is a JSON Object
"body": {
"flower": "rose",
"animal": "dog"
},
// If the body should be marshaled in a special way, you can define this here. Is not a required attribute. Standart is to marshal the body as json. Possible: [multipart,urlencoded, file]
"body_type": "urlencoded"
// If body_type is file, "body_file" points to the file to be sent as binary body
"body_file": "<path|url>"
},
// Define how the response should look like. Testtool checks against this response
"response": {
// Expected http status code. See api documentation vor the right ones
"statuscode": 200,
// If you expect certain response headers, you can define them here. A single key can have multiple headers (as defined in rfc2616)
"header": {
"key1": [
"val1",
"val2",
"val3"
],
// Headers sharing the same key are concatenated using ";", if the comparison value is a simple string,
// thus "key1" can also be checked like this:
"key1": "val1;val2;val3"
// :control in header is always applied to the flat format
"key1:control": {
// see below, this is not applied against the array
},
"x-easydb-token": [
"csdklmwerf8ßwji02kopwfjko2"
]
},
// Cookies will be under this key, in a map name => cookie
"cookie": {
"sess": {
"name": "sess",
"value": "myauthtoken"
},
"jwtoken": {
"name": "jwtoken",
"value": "tokenized",
"path": "/auth",
"domain": "mydomain",
"expires": "2021-11-10T10:00:00Z",
"max_age": 86400,
"secure": false,
"http_only": true,
"same_site": 1
}
}
// optionally, the expected format of the response can be specified so that it can be converted into json and can be checked
"format": {
"type": "csv",
"csv": {
"comma": ";"
}
},
// The body we want to assert on
"body": {
"objecttypes": [
"pictures"
]
}
},
// Store parts of the response into the datastore
"store_response_gjson": {
"eas_id": "body.0.eas._id",
// Cookies are stored in `cookie` map
"sess_cookie": "cookie.sess"
},
// wait_before_ms pauses right before sending the test request <n> milliseconds
"wait_before_ms": 0,
// wait_after_ms pauses right before sending the test request <n> milliseconds
"wait_after_ms": 0,
// Delay the request by x msec
"delay_ms": 5000,
// With the poll we can make the testing tool redo the request to wait for certain events (Only the timeout_msec is required)
// timeout_ms:* If this timeout is done, no new redo will be started
// -1: No timeout - run endless
// break_response: [Array] [Logical OR] If one of this responses occures, the tool fails the test and tells it found a break repsponse
// collect_response: [Array] [Logical AND] If this is set, the tool will check if all reponses occure in the response (even in different poll runs)
"timeout_ms": 5000,
"break_response": [
"@break_response.json"
],
"collect_response": [
"@continue_response_pending.json",
"@continue_response_processing.json"
],
// If set to true, the test case will consider its failure as a success, and the other way around
"reverse_test_result": false
}
Override template delimiters
Go template delimiters can be redefined as part of a single line comment in any of these syntax:
// template-delims: <delim_left> <delim_right>
/* template-delims: <delim_left> <delim_right> */
Examples:
// template-delims: /* */
/* template-delims: // // */
// template-delims {{ }}
/* template-delims: {* *} */
All external tests/requests/responses inherit those delimiters if not overriden in their template.
Remove template 'placeholders'
Go templates may break the proper JSONC format even when separators are comments. So we could use placeholders for filling missing parts then strip them.
// template-remove-tokens: <token> [<token>]*
/* template-remove-tokens: <token> [<token>] */
Example:
// template-delims: /* */
// template-remove-tokens: "delete_me"
{
"prop": /* datastore "something" */"delete_me"
}
This would be an actual proper JSONC as per the "delete_me"
string. However that one will be stripped before parsing the template, which would be just:
{
"prop": /* datastore "something" */
}
Unlike with delimiters, external tests/requests/responses don't inherit those removals, and need to be specified per file.
Run tests in parallel
The tool is able to run tests in parallel to themselves. You activate this mechanism by including an external test file with N@pathtofile.json
, where N
is the number of parallel "clones" you want to have of the included tests.
The included tests themselves are still run serially, only the entire set of tests will run in parallel for the specified number of replications.
This is useful e.g. for stress-testing an API.
Only tests directly included by a manifest are allowed to run in parallel.
Using "0@file.json"
will not run that specific test.
{
"name": "Example Manifest",
"tests": [
"@setup.json",
"123@foo.json",
"@cleanup.json"
]
}
Binary data comparison
The tool is able to do a comparison with a binary file. Here we take a MD5 hash of the file and and then later compare that hash.
For comparing a binary file, simply point the response to the binary file:
{
"name": "Binary Comparison",
"request": {
"endpoint": "suggest",
"method": "GET"
},
// Path to binary file with @
"response": {
"format": {
"type": "binary"
},
"body": {
"md5sum": {{ md5sum "@simple.bin" || marshal }}
}
}
}
The format must be specified as "type": "binary"
XML Data comparison
If the response format is specified as "type": "xml"
or "type": "xml2"
, we internally marshal that XML into json using github.com/clbanning/mxj.
The format "xml"
uses NewMapXmlSeq()
, whereas the format "xml2"
uses NewMapXml()
, which provides a simpler json format.
See also template file_xml2json
.
On that json you can work as you are used to with the json syntax. For seeing how the converted json looks you can use the --log-verbose
command line flag
HTML Data comparison
If the response format is specified as "type": "html"
, we internally marshal that HTML into json using github.com/PuerkitoBio/goquery.
This marshalling is less strict than for XHTML. For example it will not raise errors for unclosed tags like <p>
or <hr>
, as well as Javascript code inside the HTML code. But it is possible that unclosed tags are missing in the resulting JSON if the tokenizer can not find a matching closing tag.
See also template file_html2json
.
XHTML Data comparison
If the response format is specified as "type": "xhtml"
, we internally marshal that XHTML into json using github.com/clbanning/mxj.
The XHTML code in the response must comply to the XHTML standard, which means it must be parsable as XML.
See also template file_xhtml2json
.
CSV Data comparison
If the response format is specified as "type": "csv"
, we internally marshal that CSV into json.
You can also specify the delimiter (comma
) for the CSV format (default: ,
):
{
"name": "CSV comparison",
"request": {
"endpoint": "export/1/files/file.csv",
"method": "GET"
},
"response": {
"format": {
"type": "csv",
"csv": {
"comma": ";"
}
},
"body": {}
}
}
Preprocessing responses
Responses in arbitrary formats can be preprocessed by calling any command line tool that can produce JSON, XML, CSV or binary output. In combination with the type
parameter in format
, non-JSON output can be formatted after preprocessing. If the result is already in JSON format, it can be checked directly.
The response body is piped to the stdin
of the tool and the result is read from stdout
. The result of the command is then used as the actual response and is checked.
To define a preprocessing for a response, add a format
object that defines the pre_process
to the response definition:
{
"response": {
"format": {
"pre_process": {
"cmd": {
"name": "...",
"args": [ ],
"output": "stdout"
}
}
}
}
}
format.pre_process.cmd.name |
(string, mandatory) | name of the command line tool |
format.pre_process.cmd.args |
(string array, optional) | list of command line parameters |
format.pre_process.cmd.output |
(string, optional) | what command output to use as result response, it can be one of exitcode , stderr or stdout (default) |
Examples
Basic usage: pipe response without changes
This basic example shows how to use the pre_process
feature. The response is piped through cat
which returns the input without any changes. This command takes no arguments.
{
"response": {
"format": {
"pre_process": {
"cmd": {
"name": "cat"
}
}
}
}
}
Advanced usage: compare binary image with local one
This example shows how to use the pre_process
feature with stderr
output. The response is the metric result of running imagemagick compare
which returns the absolute error between 2 images given a threshold (0 if identical, number of different pixels otherwise). The arguments are the piped binary from the response and the image to compare against (local file using file_path
template function) .
{
"response": {
"format": {
"pre_process": {
"cmd": {
"name": "compare",
"args": [
"-metric",
"AE",
"-fuzz",
"2%",
"-",
{{ file_path "other/file.jpg" | marshal }},
"/dev/null"
],
"output": "stderr"
}
}
},
"body": 0
}
}
format.pre_process
:- Command:
compare -metric AE -fuzz 2% - /path/to/other/file.jpg /dev/null
- Parameters:
-metric AE
: metric to use for image comparison-fuzz 2%
: threshold for allowed pixel color difference-
: read first image fromstdin
instead loading a saved file/path/to/other/file.jpg
: read second image from local path (result from template function above)/dev/null
: discard stdout (it contains a binary we don't want, we use stderr output)
- Command:
Reading metadata from a file (JSON Format)
To check the file metadata of a file that is directly downloaded as a binary file using the eas/download
API, use exiftool
to read the file and output the metadata in JSON format.
If there is a file with the asset ID 1
, and the apitest needs to check that the MIME type is image/jpeg
, create the following test case:
{
"request": {
"endpoint": "eas/download/1/original",
"method": "GET"
},
"response": {
"format": {
"pre_process": {
"cmd": {
"name": "exiftool",
"args": [
"-j",
"-g",
"-"
]
}
}
},
"body": [
{
"File": {
"MIMEType": "image/jpeg"
}
}
]
}
}
format.pre_process
:- Command:
exiftool -j -g -
- Parameters:
-j
: output in JSON format-g
: group output by tag class-
: read fromstdin
instead loading a saved file
- Command:
Reading metadata from a file (XML Format)
This example shows the combination of pre_process
and type
. Instead of calling exiftool
with JSON output, it can also be used with XML output, which then will be formatted to JSON by the apitest tool.
{
"request": {
"endpoint": "eas/download/1/original",
"method": "GET"
},
"response": {
"format": {
"pre_process": {
"cmd": {
"name": "exiftool",
"args": [
"-X",
"-"
]
}
},
"type": "xml"
},
"body": [
{
"File": {
"MIMEType": "image/jpeg"
}
}
]
}
}
format.pre_process
:- Command:
exiftool -X -
- Parameters:
-X
: output in XML format-
: read fromstdin
instead loading a saved file
- Command:
format.type
:xml
: convert the output ofexiftool
, which is expected to be in XML format, into JSON
Error handling
If there is any error during the call of the command line tool, the error is formatted as a JSON object and returned instead of the expected response:
{
"command": "cat --INVALID",
"error": "exit status 1",
"exit_code": 1,
"stderr": "cat: unrecognized option '--INVALID'\nTry 'cat --help' for more information.\n"
}
command |
the command that was executed (consisting of cmd.name and cmd.args ) |
error |
error message (message of internal exec.ExitError ) |
exit_code |
integer value of the exit code |
stderr |
additional error information from stderr of the command line tool |
If such an error is expected as a result, this formatted error message can be checked as the response.
Datastore
The datastore is a storage for arbitrary data. It can be set directly or set using values received from a response. It has two parts:
- Custom storage with custom key
- Sequential response store per test suite (one manifest)
The custom storage is persistent throughout the apitest run, so all requirements, all manifests, all tests. Sequential storage is cleared at the start of each manifest.
The custom store uses a string as index and can store any type of data.
Array: If an key ends in []
, the value is assumed to be an Array, and is appended. If no Array exists, an array is created.
Map: If an key ends in [key]
, the value is assumed to be an map, and writes the data into the map at that key. If no map exists, an map is created.
{
"store": {
"eas_ids[]": 15,
"mapStorage[keyIWantToStore]": "value"
}
}
This example would create an Array in index eas_ids and append 15 to it.
Arrays are useful using the Go-Template range function.
Set Data in Custom Store
To set data in custom store, you can use 4 methods:
store on the manifest.json top level |
the data is set before the session authentication (if any) |
store_response_gjson in authentication.store_response_gjson |
|
store on the test level |
the data is set before request and response are evaluated |
store_response_gjson on the test level |
the data is set after each response (If you want the datestore to delete the current entry if no new one could be found with gjson . Just prepend the gjson key with a ! . E.g. "eventId":"!body.0._id" will delete the eventId entry from the datastore if body.0._id could not be found in the response json) |
All methods use a Map as value, the keys of the map are string, the values can be anything. If the key (or index) ends in []
and Array is created if the key does not yet exist, or the value is appended to the Array if it does exist.
The method store_response_gjson
takes only string as value. This gjson
-string is used to parse the current response using the gjson
feature. The return value from the gjson
call is then stored in the datastore.
Get Data from Custom Store
The data from the custom store is retrieved using the datastore <key>
Template function. key
must be used in any store method before it is requested. If the key is unset, the datastore function returns an empty string. Use the special key -
to return the entire datastore.
Slices allow the backwards index access. If you have a slice of length 3 and access it at index -1
you get the last element in the slice (original index 2
)
If you access an invalid index for datastore map[index]
or slice[]
you get an empty string. No error is thrown.
Get Data from Sequential Store
To get the data from the sequential store an integer number has to be given to the datastore function as string. So datastore "0"
would be a valid request. This would return the response from first test of the current manifest. datastore "-1"
returns the last response from the current manifest. datastore "-2"
returns second to last from the current manifest. If the index is wrong the function returns an error.
The sequential store stores the body and header of all responses. Use gjson
to access values in the responses. See template functions datastore
and gjson
.
When using relative indices (negative indices), use the same index to get values from the datastore to use in the request and response definition. Especially, for evaluating the current response, it has not yet been stored. So, datastore "-1"
will still return the last response in the datastore. The current response will be appended after it was evaluated, and then will be returned with datastore "-1"
.
Use control structures
We support certain control structures in the response definition. You can use this control structures when ever you are able to set keys in the json (so you have to be inside a object).
Some of them also need a value and some don't. For those which don't need a value you can just setup the control structure without a second key with some weird value. When you give a value the tool always tries to deep check if that value is correct and present in the actual reponse. So be aware of this behavior as it could interfere with your intended test behavior.
Define a control structure
In the example we use the jsonObject test
and define some control structures on it. A control structure uses the key it is attached to plus :control
. So for our case it would be test:control
. The tool gets that this two keys test
and test:control
are in relationship with each other.
{
"test": {
"hallo": 2,
"hello": 3
},
"test:control": {
"is_object": true,
"no_extra": true
}
}
body:control
All controls, which are defined below, can also be applied to the complete response body itself by setting body:control
. The control check functions work the same as on any other key. This can be combined with other controls inside the body.
Available controls
There are several controls available. The first two no_extra
and order_matters
always need their responding real key and value to function as intended. The others can be used without a real key.
Default behavior for all keys is false
. So you only have to set them if you want to explicit use them as true
.
no_extra
This command defines an exact match. If it is set, there are no more fields allowed in the response as defined in the testcase.
no_extra
is available for objects and arrays.
The following response would fail as there are to many fields in the actual response:
expected response defined with no_extra
{
"body": {
"testObject": {
"a": "z",
"b": "y"
},
"testObject:control": {
"no_extra": true
}
}
}
actual response
{
"body": {
"testObject": {
"a": "z",
"b": "y",
"c": "to much, so we fail"
}
}
}
order_matters
This command defines that the order in an array should be checked.
order_matters
is available only for arrays
E.g. the following response would fail as the order in the actual response is wrong:
expected response defined with order_matters
{
"body": {
"testArray": [
"a",
"b",
"c"
],
"testArray:control": {
"order_matters": true
}
}
}
actual response
{
"body": {
"testArray": [
"c",
"b",
"a"
]
}
}
depth
This setting defines the depth that the no_extra
and order_matters
should consider when matching arrays.
depth
is available only for arrays.
The possible values of depth
are:
-1 |
full depth |
0 |
top element only (default) |
N |
N elements deep |
The following response would fail as there are too many entries in the actual response inner array:
expected response defined with no_extra
and depth
{
"body": {
"testArray": [
[1, 3, 5],
[2, 4, 6]
],
"testObject:control": {
"no_extra": true,
"depth": 1
}
}
}
actual response
{
"body": {
"testArray": [
[1, 3, 5],
[2, 4, 6, 8]
]
}
}
must_exist
Check if a certain value does exist in the reponse (no matter what its content is).
must_exist
is available for all types.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "iShouldExist"
is not in the actual response:
expected response defined with must_exist
{
"body": {
"iShouldExist:control": {
"must_exist": true
}
}
}
actual response
{
"body": {}
}
element_count
Check if the size of an array equals the element_count
.
element_count
is available only for arrays.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "count"
has the wrong length:
expected response defined with element_count
{
"body": {
"count:control": {
"element_count": 2
}
}
}
actual response
{
"body": {
"count": [
1,
2,
3
]
}
}
element_no_extra
Passes the no_extra
check to the underlying structure in an array.
element_no_extra
is only available for arrays.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "extra"
has an extra element:
expected response defined with element_no_extra
{
"body": {
"count": [
{
"fine": true,
}
],
"count:control": {
"element_no_extra": true
}
}
}
actual response
{
"body": {
"count": [
{
"fine": true,
"extra": "shouldNotBeHere"
}
]
}
}
must_not_exist
Check if a certain value does not exist in the reponse.
must_not_exist
is available for all types.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "iShouldNotExist"
is in the actual response:
expected response defined with must_not_exist
{
"body": {
"iShouldNotExist:control": {
"must_not_exist": true
}
}
}
{
"body": {
"iShouldNotExist": "i exist, hahahah"
}
}
not_equal
Check if a field is not equal to a specific value.
This check is available for the types string
, number
, array
and bool
.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testNumber"
has the value 5
:
expected response defined with not_equal
{
"body": {
"testNumber:control": {
"not_equal": 5
}
}
}
actual response
{
"body": {
"testNumber": 5
}
}
match
Check if a string value matches a given regular expression
E.g. the following response would fail as "text"
does not match the regular expression:
expected string response checked with a regex:
{
"body": {
"text:control": {
"match": ".+-\\d+"
}
}
}
actual response
{
"body": {
"text": "valid_string-123"
}
}
not_match
Check if a string value does not match a given regular expression.
This is the opposite check function of match.
starts_with
Check if a string value starts with a given string prefix.
E.g. the following response would fail as "text"
does not have the prefix:
expected string response checked with a prefix
{
"body": {
"text:control": {
"starts_with": "abc-"
}
}
}
actual response
{
"body": {
"text": "abc-123"
}
}
ends_with
Check if a string value ends with a given string suffix.
E.g. the following response would fail as "text"
does not have the suffix:
expected string response checked with a suffix
{
"body": {
"text:control": {
"ends_with": "-123"
}
}
}
actual response
{
"body": {
"text": "abc-123"
}
}
is_string
Check if the field has the type string
.
It implicitly also checks must_exist
for the value as there is no sense in type checking a value that does not exist.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testString"
is not a string in the actual response:
expected response defined with is_string
{
"body": {
"testString:control": {
"is_string": true
}
}
}
actual response
{
"body": {
"testString": 555
}
}
is_bool
Check if the field has the type bool
.
It implicitly also checks must_exist
for the value.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testBool"
is no boolean value in the actual response:
expected response defined with is_bool
{
"body": {
"testBool:control": {
"is_bool": true
}
}
}
actual response
{
"body": {
"testBool": "not a boolean"
}
}
is_number
Check if the field has the type number
.
It implicitly also checks must_exist
for the value.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testNumber"
is no numeric value in the actual response:
expected response defined with is_number
{
"body": {
"testNumber:control": {
"is_number": true
}
}
}
actual response
{
"body": {
"testNumber": "not a number"
}
}
is_object
Check if the field is a JSON object.
It implicitly also checks must_exist
for the value.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testObj"
is not an object in the actual response:
expected response defined with is_object
{
"body": {
"testObj:control": {
"is_object": true
}
}
}
actual response
{
"body": {
"testObj": "not an object"
}
}
is_array
Check if the field is a JSON array.
It implicitly also checks must_exist
for the value.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "testArr"
is not an array in the actual response:
expected response defined with is_array
{
"body": {
"testArr:control": {
"is_array": true
}
}
}
actual response
{
"body": {
"testArr": "not an array"
}
}
number_gt
With number_gt
(>
), you can check if your field of type number (implicit check) is greater than a specific number.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "beGreater"
is equal to the expected number:
expected response defined with number_gt
{
"body": {
"beGreater:control": {
"number_gt": 5
}
}
}
actual response
{
"body": {
"beGreater": 5
}
}
number_ge
With number_ge
(=>
), you can check if your field of type number (implicit check) is equal or greater than a specific number.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "beGreaterOrEqual"
is less than the expected number:
expected response defined with number_ge
{
"body": {
"beGreaterOrEqual:control": {
"number_ge": 5
}
}
}
actual response
{
"body": {
"beGreaterOrEqual": 3
}
}
number_lt
With number_lt
(<
), you can check if your field of type number (implicit check) is less than a specific number.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "beLess"
is equal to the expected number:
expected response defined with number_lt
{
"body": {
"beLess:control": {
"number_lt": 5
}
}
}
actual response
{
"body": {
"beLess": 5
}
}
number_le
With number_le
(<=
), you can check if your field of type number (implicit check) is less or equal than a specific number.
This control can be used without the actual key. So only the :control
key is present.
E.g. the following response would fail as "beLessOrEqual"
is greater than the expected number:
expected response defined with number_le
{
"body": {
"beLessOrEqual:control": {
"number_le": 5
}
}
}
actual response
{
"body": {
"beLessOrEqual": 7
}
}
Use external file
In the request and response part of the single testcase you also can load the content from an external file.
This is exspecially helpfull for keeping the manifest file simpler/smaller and keep a better overview. On top: You can use so called template functions in the external file. (We will dig deeper into the template functions later)
A single test could look as simple as following:
{
"name": "Test loading request & response from external file",
"request": "@path/to/requestFile.json",
"response": "@path/to/responseFile.json"
}
Important: The paths to the external files start with a @
and are relative to the location of the manifest.json
or can be web urls e.g. https://programmfabrik.de/testfile.json
The content of the request and response file are execatly the same as if you would place the json code inline:
Request:
{
"body": {
"animal": "dog",
"flower": "rose"
},
"body_type": "urlencoded",
"endpoint": "suggest",
"header": {
"header1": "value",
"header2": "value"
},
"method": "GET",
"query_params": {
"number": 2,
"token": "testtoken"
}
}
Response:
{
"body": {
"objecttypes": [
"pictures"
],
"query": ">>>[0-9]*<<<"
},
"header": {
"key1": [
"val1",
"val2",
"val3"
],
"x-easydb-token": [
"csdklmwerf8\u00dfwji02kopwfjko2"
]
},
"statuscode": 200
}
Template functions
apitest supports the Sprig template function library in v3. Internally provided functions like add
overwrite the Sprig
function.
As described before, if you use an external file you can make use of so called template functions. What they are and how they work for the apitesting tool is described in the following part.
Template Functions are invoked using the tags {{ }}
and upon returning substitutes the function call with its result. We use the golang "text/template" package so all functions provided there are also supported here.
For a reference see https://golang.org/pkg/text/template
manifest.json -> external file |
load external file |
external file -> another file | render template with file parameter "hello" |
another file -> external file | return rendered template "hello world" |
external file -> manifest.json |
return rendered template |
Example
Assume that the template function myfunc
, given the arguments 1 "foo"
, returns "bar"
. The call {{ myfunc 1 "foo" }}
would translate to bar
. Consequently, rendering Lets meet at the {{ myfunc 1 "foo" }}
results in an invitation to the bar
.
We provide the following functions:
file_render "relative/path/" [param, ...]
Helper function to load contents of a file; if this file contains templates; it will render these templates with the parameters provided in the can be accessed from the loaded file via {{ .Param1-n }};
see example below
Loads the file with the relative path ( to the file this template function is invoked in ) "relative/path" or a weburl e.g. https://docs.google.com/test/tmpl.txt. Returns string.
file "relative/path/"
Loads the file with the relative path ( to the file this template function is invoked in ) "relative/path" or a weburl e.g. https://docs.google.com/test/tmpl.txt. Returns string.
Example
Content of file at some/path/example.tmpl
:
{{ load_file "../target.tmpl" "hello" }}
Content of file at some/target.tmpl
:
{{ .Param1 }} world
Rendering example.tmpl
will result in hello world
file_path "relative/path/"
Returns the relative path (to the file this template function is invoked in) "relative/path" or a weburl e.g. https://docs.google.com/test/tmpl.txt
Example
Absolute path of file at some/path/myfile.cpp
:
{{ file_path "../myfile.tmpl" }}
pivot_rows
"keyColumn" "typeColumn" [input]
Read a CSV map and turn rows into columns and columns into rows.
Assume you have the following structure in your sheet:
key | type | 1 | 2 |
string | string | string | string |
name | string | bicyle | car |
wheels | int64 | 2 | 4 |
As a convention the data columns need to be named 1
, 2
, ... Allowed types are:
string
int64
number
(JSON type number)float64
Calling
pivot_rows("key","type",(file_csv "file.csv" ','))
returns
[
{
"filename": "bicyle",
"wheels": 2
},
{
"filename": "car",
"wheels": 4
}
]
rows_to_map "keyColumn" "valueColumn" [input]
Generates a key-value map from your input rows.
Assume you have the following structure in your sheet:
column_a | column_b | column_c |
---|---|---|
row1a | row1b | row1c |
row2a | row2b | row2c |
If you parse this now to CSV and then load it via file_csv
you get the following JSON structure:
[
{
"column_a": "row1a",
"column_b": "row1b",
"column_c": "row1c"
},
{
"column_a": "row2a",
"column_b": "row2b",
"column_c": 22
}
]
For mapping now certain values to a map you can use rows_to_map "column_a" "column_c"
and the output will be a map with the following content:
{
"row1a": "row1c",
"row2a": 22
}
group_rows "groupColumn" [rows]
Generates an Array of rows from input rows. The groupColumn needs to be set to a column which will be used for grouping the rows into the Array.
The column needs to:
- be an int64 column
- use integers between
0
and999
The Array will group all rows with identical values in the groupColumn.
Example
The CSV can look at follows, use file_csv to read it and pipe into group_rows
batch | reference | title |
---|---|---|
int64 | string | string |
1 | ref1a | title1a |
1 | ref1b | title1b |
4 | ref4 | title4 |
3 | ref3 | titlte2 |
Produces this output (presented as json for better readability:
[
[
{
"batch": 1,
"reference": "ref1a",
"title": "title1a"
},
{
"batch": 1,
"reference": "ref1b",
"title": "title1b"
}
],
[
{
"batch": 3,
"reference": "ref3",
"title": "title3"
}
],
[
{
"batch": 4,
"reference": "ref4",
"title": "title4"
}
]
]
group_map_rows "groupColumn" [rows]
Generates an Map of rows from input rows. The groupColumn needs to be set to a column which will be used for grouping the rows into the Array.
The column needs to be a string column.
The Map will group all rows with identical values in the groupColumn.
Example
The CSV can look at follows, use file_csv to read it and pipe into group_rows
batch | reference | title |
---|---|---|
string | string | string |
one | ref1a | title1a |
one | ref1b | title1b |
4 | ref4 | title4 |
3 | ref3 | titlte2 |
Produces this output (presented as json for better readability:
{
"one": [
{
"batch": "one",
"reference": "ref1a",
"title": "title1a"
},
{
"batch": "one",
"reference": "ref1b",
"title": "title1b"
}
],
"4": [
{
"batch": "4",
"reference": "ref3",
"title": "title3"
}
],
"3": [
{
"batch": "3",
"reference": "ref4",
"title": "title4"
}
]
}
Template Example
With the parameters keyColumn
and valueColumn
you can select the two columns you want to use for map. (Only two are supported)
The keyColumn
must be of the type string, as it functions as map index (which is of type string)
{{ unmarshal "[{\"column_a\": \"row1a\",\"column_b\": \"row1b\",\"column_c\": \"row1c\"},{\"column_a\": \"row2a\",\"column_b\": \"row2b\",\"column_c\": \"row2c\"}]" | rows_to_map "column_a" "column_c" | marshal }}
Rendering that will give you :
{
"row1a": "row1c",
"row2a": "row2c"
}
Behavior in corner cases
No keyColumn given
The function returns an empty map
For rows_to_map
:
{}
No valueColumn given
The complete row gets mapped
For rows_to_map "column_a"
:
{
"row1a": {
column_a: "row1a",
column_b: "row1b",
column_c: "row1c",
},
"row2a": {
column_a: "row2a",
column_b: "row2b",
column_c: "row2c",
}
}
A row does not contain a key column
The row does get skipped
Input:
[
{
column_a: "row1a",
column_b: "row1b",
column_c: "row1c",
},
{
column_b: "row2b",
column_c: "row2c",
}
{
column_a: "row3a",
column_b: "row3b",
column_c: "row3c",
}
]
For rows_to_map "column_a" "column_c"
:
{
row1a: "row1c",
row3a: "row3c",
}
A row does not contain a value column
The value will be set to ""
(empty string)
Input:
[
{
column_a: "row1a",
column_b: "row1b",
column_c: "row1c",
},
{
column_a: "row2a",
column_b: "row2b",
}
{
column_a: "row3a",
column_b: "row3b",
column_c: "row3c",
}
]
For rows_to_map "column_a" "column_c"
:
{
"row1a": "row1c",
"row2a": "",
"row3a": "row3c",
}
datastore [key]
Helper function to query the datastore; used most of the time in conjunction with gjson
.
The key
can be an int, or int64 accessing the store of previous responses. The responses are accessed in the order received. Using a negative value access the store from the back, so a value of -2 would access the second to last response struct.
This function returns a string, if the key
does not exist, an empty string is returned.
If the key
is a string, the datastore is accessed directly, allowing access to custom set values using store
or store_response_gjson
parameters.
The datastore stores all responses in a list. We can retrieve the response (as a json string) by using this template function. {{ datastore 0 }}
will render to
{
"statuscode": 200,
"header": {
"foo": "bar;baz"
},
"body": "..."
}
This function is intended to be used with the gjson
template function.
The key -
has a special meaning, it returns the entire custom datastore (not the sequentially stored responses)
gjson [path] [json]
Helper function to extract fields from the json
. It uses gjson
syntax. For more information, see the external documentation.
Parameter | Type | Description |
---|---|---|
@path |
string |
a description of the location of the field to extract. For array access use integers; for object access use keys. Example: body.1.field ; see below for more details |
@json_string |
string |
a valid json blob to be queried; can be supplied via pipes from datastore idx |
@result |
the content of the json blob at the specified path |
Example
The call
{{ gjson "foo.1.bar" "{\"foo": [{\"bar\": \"baz\"}, 42]}" }}
would return baz
.
As an example with pipes, the call
{{ datastore idx | gjson "header.foo.1" }}
would returnbar
given the response above.
file_csv [path] [delimiter]
Helper function to load a csv file
Parameter | Type | Description |
---|---|---|
@path |
string |
A path to the csv file that should be loaded. The path is either relative to the manifest or a weburl |
@delimiter |
rune |
The delimiter that is used in the given csv e.g. , Defaults to , |
@result |
The content of the csv as json array so we can work on this data with gjson |
The CSV must have a certain structur. If the structure of the given CSV differs, the apitest tool will fail with a error
- In the first row must be the names of the fields
- In the seconds row must be the types of the fields
Valid types
int64
int
string
float64
bool
int64,array
string,array
float64,array
bool,array
json
All types can be prefixed with *
to return a pointer to the value. Empty strings initialize the Golang zero value for the type, for type array the empty string inialized an empty array. The empty string returns an untyped nil.
Example
Content of file at some/path/example.csv
:
id,name
int64,string
1,simon
2,martin
The call
{{ file_csv "some/path/example.csv" ','}}
would result in
[map[id:1 name:simon] map[id:2 name:martin]]
As an example with pipes, the call
{{ file_csv "some/path/example.csv" ',' | marshal | gjson "1.name" }}
would result in martin
given the response above.
Corner Cases
There are some corner cases that trigger a certain behavior you should keep in mind
No format for a column given
The column gets skipped in every row
Input
id,name
int64,
1,simon
2,martin
Result
[map[id:1] map[id:2]]
No name for a column given
The column gets skipped in every row
Input
,name
int64,string
1,simon
2,martin
Result
[map[name:simon] map[name:martin]]
Comment or empty line
If there is a comment marked with #
, or a empty line that does not get rendered into the result
Input
id,name
int64,string
1,simon
2,martin
#3,philipp
4,roman
#5,markus
6,klaus
7,sebastian
Result
[map[name:simon] map[name:martin] map[name:roman] map[name:klaus] map[name:sebastian]]
file_xml2json [path]
Helper function to parse an XML file and convert it into json
@path
: string; a path to the XML file that should be loaded. The path is either relative to the manifest or a weburl
This function uses the function NewMapXml()
from github.com/clbanning/mxj.
Example
Content of XML file some/path/example.xml
:
<objects xmlns="https://schema.easydb.de/EASYDB/1.0/objects/">
<obj>
<_standard>
<de-DE>Beispiel Objekt</de-DE>
<en-US>Example Object</en-US>
</_standard>
<_system_object_id>123</_system_object_id>
<_id>45</_id>
<name type="text_oneline"
column-api-id="263">Example</name>
</obj>
</objects>
The call
{{ file_xml2json "some/path/example.xml" }}
would result in
{
"objects": {
"-xmlns": "https://schema.easydb.de/EASYDB/1.0/objects/",
"obj": {
"_id": "45",
"_standard": {
"de-DE": "Beispiel Objekt",
"en-US": "Example Object"
},
"_system_object_id": "123",
"name": {
"#text": "Example",
"-column-api-id": "263",
"-type": "text_oneline"
}
}
}
}
file_html2json [path]
Helper function to parse an HTML file and convert it into json
@path
: string; a path to the HTML file that should be loaded. The path is either relative to the manifest or a weburl
This marshalling is less strict than for XHTML. For example it will not raise errors for unclosed tags like <p>
or <hr>
, as well as Javascript code inside the HTML code. But it is possible that unclosed tags are missing in the resulting JSON if the goquery tokenizer can not find a matching closing tag.
Example
Content of HTML file some/path/example.html
:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<title>fylr</title>
<meta name="description" content="fylr - manage your data" />
<script>
function onInputHandler(event) {
const form = event.currentTarget;
submitForm(form);
}
</script>
</head>
<body>
<div class="container">
<h1>Register</h1>
<p class="required-information"><sup>*</sup>Mandatory fields<br>
<p class="error-summary">Form has errors
<hr>
</div>
</body>
</html>
The call
{{ file_html2json "some/path/example.html" }}
would result in
{
"html": {
"-lang": "en",
"head": {
"meta": [
{
"-charset": "utf-8"
},
{
"-content": "fylr - manage your data",
"-name": "description"
}
],
"title": {
"#text": "fylr"
},
"script": {
"#text": "function onInputHandler(event) {\n\t\t\t\tconst form = event.currentTarget;\n\t\t\t\tsubmitForm(form);\n\t\t\t}"
}
},
"body": {
"div": {
"-class": "container",
"h1": {
"#text": "Register"
},
"p": [
{
"-class": "required-information",
"sup": {
"#text": "*"
},
"br": {}
},
{
"#text": "Form has errors",
"-class": "error-summary"
}
],
"hr": {}
}
}
}
}
file_xhtml2json [path]
Helper function to parse an XHTML file and convert it into json
@path
: string; a path to the XHTML file that should be loaded. The path is either relative to the manifest or a weburl
Example
Content of XHTML file some/path/example.xhtml
:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<link href="/css/easydb.css" rel="stylesheet" type="text/css" />
<title>easydb documentation</title>
</head>
<body>
<h1 id="welcome-to-the-easydb-documentation">Welcome to the easydb documentation</h1>
</body>
</html>
The call
{{ file_xhtml2json "some/path/example.xhtml" }}
would result in
{
"html": {
"-xmlns": "http://www.w3.org/1999/xhtml",
"head": {
"link": {
"-href": "/css/easydb.css",
"-rel": "stylesheet",
"-type": "text/css"
},
"title": "easydb documentation"
},
"body": {
"h1": {
"#text": "Welcome to the easydb documentation",
"-id": "welcome-to-the-easydb-documentation"
}
}
}
}
file_sqlite [path] [statement]
Helper function to return the result of an SQL statement from a sqlite3 file
Parameter | Type | Description |
---|---|---|
@path |
string | a path to the sqlite file that should be loaded. The path is either relative to the manifest or a weburl |
@statement |
string | a SQL statement that returns data (SELECT ) |
@result |
the result of the statement as a json array so we can work on this data with gjson |
Example
Content of sqlite file at some/path/example.sqlite
:
Table names
:
- column
id
: typeINTEGER
- column
name
: typeTEXT
id | name |
---|---|
2 |
martin |
3 |
NULL |
1 |
simon |
The call
{{ file_sqlite "some/path/example.sqlite" `
SELECT id, name FROM names
WHERE name IS NOT NULL
ORDER BY id ASC
` }}
would result in
[map[id:1 name:simon] map[id:2 name:martin]]
Working with NULL
values
NULL
values in the database are returned as nil
in the template. To check if a value in the sqlite file is NULL
, us a comparison to nil
:
The call
{{ file_sqlite "some/path/example.sqlite" `
SELECT id, name FROM names
ORDER BY id ASC
` }}
would result in
[map[id:1 name:simon] map[id:2 name:martin] map[id:3 name:nil]]
The NULL
value in name
can be checked with
{{ if ne $row.name nil }}
// use name, else skip
{{ end }}
slice [parms...]
Returns a slice with the given parameters as elements. Use this for range in templates.
split s sep
Returns a string slice with s
split by sep
.
add [a] [b]
Returns the sum of a
and b
. a, b
can be any numeric type or string. The function returns a numeric type, depending on the input. With string
we return int64
.
subtract [a] [b]
Returns a - b
. a, b
can be any numeric type or string. The function returns a numeric type, depending on the input. With string
we return int64
.
multiply [a] [b]
Returns a * b
. a, b
can be any numeric type or string. The function returns a numeric type, depending on the input. With string
we return int64
.
divide [a] [b]
Returns a / b
. a, b
can be any numeric type or string. The function returns a numeric type, depending on the input. With string
we return int64
.
unmarshal [string]
Returns a util.GenericJson
Object (go: interface{}
) of the unmarshalled JSON
string.
marshal [interface{}]
Returns a string
of the marshalled interface{}
object.
md5sum [filepath]
Returns a string
of the MD5 sum of the file found in filepath
.
str_escape [string]
Returns a string
where all "
are escaped to \"
. This is useful in Strings which need to be concatenated.
query_escape [string]
Returns a string
as the result of escaping input as if it was intended for use in a URL query string.
query_unescape [string]
Returns a string
as the result of unescaping input as if it was coming from a URL query string.
base64_encode [string]
Returns a string
as the result of encoding input into base64.
base64_decode [string]
Returns a string
as the result of decoding input from base64.
url_path_escape [string]
Uses Url.PathEscape to escape given string
to use in endpoint
or server_url
. Returns string
.
match [regex] [text]
Returns a bool
value. If text
matches the regular expression regex
, it returns true
, else false
. This is useful inside {{ if ... }}
templates.
printf [interface{}...]
Just for reference, this is a Go Template built-in.
N [float64|int64|int]
Returns a slice of n 0-sized elements, suitable for ranging over.
Example how to range over 100 objects
{
"body": [
{{ range $idx, $v := N 100 }}
...
{{ end }}
]
}
replace_host [url]
replace_host replaces the host in the given url
with the actual address of the built-in HTTP server (see below). This address, taken from the manifest.json
can be overwritten with the command line parameter --replace-host
.
As an example, the URL http://localhost/myimage.jpg
would be changed into http://localhost:8788/myimage.jpg
following the example below.
server_url
server_url returns the server url, which can be globally provided in the config file or directly by the command line parameter --server
. This is a *url.URL
.
server_url_no_user
server_url_no_user returns the server url, which can be globally provided in the config file or directly by the command line parameter --server
. Any information about the user authentification is removed. This is a *url.URL
.
If the server_url is in the form of http://user:password@localhost
, server_url_no_user will return http://localhost
.
is_zero
is_zero returns true if the passed value is the Golang zero value of the type.
oauth2_password_token [client] [username] [password]
oauth2_password_token returns an oauth token for a configured client and given some user credentials. Such token is an object which contains several properties, being access_token one of them. It uses the trusted
oAuth2 flow
Example:
{
"store": {
"access_token": {{ oauth2_password_token "my_client" "john" "pass" | marshal | gjson "access_token" }}
}
}
oauth2_client_token [client]
oauth2_client_token returns an oauth token for a configured client. Such token is an object which contains several properties, being access_token one of them. It uses the client credentials
oAuth2 flow.
Example:
{
"store": {
"access_token": {{ oauth2_client_token "my_client" | marshal | gjson "access_token" }}
}
}
oauth2_code_token [client] ...[[key] [value]]
oauth2_code_token returns an oauth token for a configured client and accepts a variable number of key/value parameters. Such token is an object which contains several properties, being access_token one of them. It uses the code grant
oAuth2 flow.
Behind the scenes the function will do a GET request to the auth URL
, adding such parameters to it, and interpret the last URL such request was redirected to, extracting the code from it and passing it to the last step of the regular flow.
Example:
{
"store": {
"access_token": {{ oauth2_code_token "my_client" "username" "myuser" "password" "mypass" | marshal | gjson "access_token" }}
}
}
Or:
{
"store": {
"access_token": {{ oauth2_code_token "my_client" "guess_access" "true" | marshal | gjson "access_token" }}
}
}
oauth2_implicit_token [client] ...[[key] [value]]
oauth2_implicit_token returns an oauth token for a configured client and accepts a variable number of key/value parameters. Such token is an object which contains several properties, being access_token one of them. It uses the implicit grant
oAuth2 flow.
Behind the scenes the function will do a GET request to the auth URL
, adding such parameters to it, and interpret the last URL such request was redirected to, extracting the token from its fragment.
Example:
{
"store": {
"access_token": {{ oauth2_password_token "my_client" "myuser" "mypass" | marshal | gjson "access_token" }}
}
}
oauth2_client [client]
oauth2_client returns a configured oauth client given its client_id
. Result is an object which contains several properties.
Example:
{
"store": {
"oauth2_client_config": {{ oauth2_client "my_client" | marshal }}
}
}
oauth2_basic_auth [client]
oauth2_basic_auth returns the authentication header for basic authentication for the given oauth client.
semver_compare [version 1] [version 2]
semver_compare compares to semantic version strings. This calls https://pkg.go.dev/golang.org/x/mod/semver#Compare, so check there for additional documentation. If the version is ""
the version v0.0.0
is assumed. Before comparing, the function checks if the strings are valid. In case they are not, an error is returned.
log
[msg] [args...]
Write msg to log output. Args can be given. This uses logrus.Debugf to output.
remove_from_url
[key] [url]
Removes from key from url's query, returns the url with the key removed. In case of an error, the url is returned as is. Unparsable urls are ignored and the url is returned.
value_from_url
[key]
Returns the value from the url's query for key. In case of an error, an empty string is returned. Unparsable urls are ignored and an empty string is returned.
parallel_run_idx
Returns the index of the Parallel Run that the template is executed in, or -1 if it is not executed within a parallel run.
HTTP Server
The apitest tool includes an HTTP Server. It can be used to serve files from the local disk temporarily. The HTTP Server can run in test mode. In this mode, the apitest tool does not run any tests, but starts the HTTP Server in the foreground, until CTRL-C in pressed. It is possible to define a proxy in the server which accepts and stores request data. It is useful if there is need to test that expected webhook calls are properly performed.
Different stores can be configured within the proxy.
To configure a HTTP Server, the manifest need to include these lines:
{
"http_server": {
"addr": ":8788", // address to listen on
"dir": "", // directory to server, relative to the manifest.json, defaults to "."
"testmode": false, // boolean flag to switch test mode on / off
"proxy": { // proxy configuration
"test": { // proxy store configuration
"mode": "passthru" // proxy store mode
}
}
}
}
The proxy mode
parameter supports these values:
passthru
: The request is stored as it is, without further processing
The HTTP Server is started and stopped per test.
HTTP Endpoints
The server provides endpoints to serve local files and return responses based on request data.
Static files
To access any static file, use the path relative to the server directory (dir
) as the endpoint:
{
"request": {
"endpoint": "path/to/file.jpg",
"method": "GET"
}
}
If there is any error (for example wrong path), a HTTP error repsonse will be returned.
No Content-Length header
For some tests, you may not want the Content-Length header to be sent alongside the asset
In this case, add no-content-length=1
to the query string of the asset url:
{
"request": {
"endpoint": "path/to/file.jpg?no-content-length=1",
"method": "GET"
}
}
bounce
The endpoint bounce
returns the binary of the request body, as well as the request headers and query parameters as part of the response headers.
{
"request": {
"endpoint": "bounce",
"method": "POST",
"query_params": {
"param1": "abc"
},
"header": {
"header1": "123"
},
"body": {
"file": "@path/to/file.jpg"
},
"body_type": "multipart"
}
}
The file that is specified is relative to the apitest file, not relative to the http server directory. The response will include the binary of the file, which can be handled with pre_process
and format
.
Request headers are included in the response header with the prefix X-Req-Header-
, request query parameters are included in the response header with the prefix X-Req-Query-
:
{
"response": {
"header": {
"X-Req-Query-Param1": [
"abc"
],
"X-Req-Header-Header1": [
"123"
]
}
}
}
bounce-json
The endpoint bounce-json
returns the a response that includes header
, query_params
and body
in the body.
{
"request": {
"endpoint": "bounce-json",
"method": "POST",
"query_params": {
"param1": "abc"
},
"header": {
"header1": 123
},
"body": {
"value1": "test",
"value2": {
"hello": "world"
}
}
}
}
will return this response:
{
"response": {
"body": {
"query_params": {
"param1": [
"abc"
]
},
"header": {
"Header1": [
"123"
]
},
"body": {
"value1": "test",
"value2": {
"hello": "world"
}
}
}
}
}
bounce-query
The endpoint bounce-query
returns the a response that includes in its body
the request query string
as it is.
This is useful in endpoints where a body cannot be configured, like oAuth urls, so we can simulate responses in the request for testing.
{
"request": {
"endpoint": "bounce-query?here=is&all=stuff",
"method": "POST",
"body": {}
}
}
will return this response:
{
"response": {
"body": "here=is&all=stuff"
}
}
HTTP Server Proxy
The proxy different stores can be used to both store and read their stored requests.
The configuration, as already defined in HTTP Server, is as follows:
"proxy": { // proxy configuration
"<store_name>": { // proxy store configuration
"mode": "passthru" // proxy store mode
}
}
Key | Value Type | Value description |
---|---|---|
proxy |
JSON Object | An object with the store names as keys and their configuration as values |
<store_name> |
JSON Object | An object with the store configuration |
mode |
string | The mode the store runs on (see below) |
Store modes:
Value | Description |
---|---|
passthru |
The request to the proxy store will be stored as it is without any further processing |
Write to proxy store
Perform a request against the http server path /proxywrite/<store_name>
. Where <store_name>
is a key (store name) inside the proxy
object in the configuration.
The expected response will have either 200
status code and the used offset as body or another status and an error body.
Given this request:
{
"endpoint": "/proxywrite/test",
"method": "POST",
"query_params": {
"some": "param"
},
"header": {
"X-My-Header": 0
},
"body": {
"post": {
"my": [
"body",
"here"
]
}
}
}
The expected response:
{
"statuscode": 200,
"body": {
"offset": 0
}
}
Read from proxy store
Whatever request performed against the server path /proxyread/<store_name>?offset=<offset>
.
Where:
<store_name>
is a key inside theproxy
object in the server configuration, aka the proxy store name<offset>
represents the entry to be retrieved in the proxy store requests collection. If not provided, 0 is assumed.
Given this request:
{
"endpoint": "/proxyread/test",
"method": "GET",
"query_params": {
"offset": 0
}
}
The expected response:
{
// Merged headers. original request headers prefixed with 'X-Request`
"header": {
// The method of the request to the proxy store
"X-Apitest-Proxy-Request-Method": [
"POST"
],
// The url path requested (including query string)
"X-Apitest-Proxy-Request-Path": [
"/proxywrite/test"
],
// The request query string only
"X-Apitest-Proxy-Request-Query": [
"is=here&my=data&some=value"
],
// Original request custom header
"X-My-Header": [
"blah"
],
// The number of requests stored
"X-Apitest-Proxy-Store-Count": [
"7"
],
// The next offset in the store
"X-Apitest-Proxy-Store-Next-Offset": [
"1"
]
... // All other standard headers sent with the original request (like Content-Type)
},
// The body of this request to the proxy store, always in binary format
"body": {
// Content-Type header will reveal its format on client side, in this case, it's JSON, but it could be a byte stream of an image, etc.
"whatever": [
"is",
"here"
]
}
}
SMTP Server
Summary and Configuration
The apitest tool can run a mock SMTP server intended to catch locally sent emails for testing purposes.
To add the SMTP Server to your test, put the following in your manifest:
{
"smtp_server": {
"addr": ":9025", // address to listen on
"max_message_size": 1000000 // maximum accepted message size in bytes
// (defaults to 30MiB)
}
}
The server will then listen on the specified address for incoming emails. Incoming messages are stored in memory and can be accessed using the HTTP endpoints described further below. No authentication is performed when receiving messages.
If the test mode is enabled on the HTTP server and an SMTP server is also configured, both the HTTP and the SMTP server will be available during interactive testing.
HTTP Endpoints
On its own, the SMTP server has only limited use, e.g. as an email sink for applications that require such an email sink to function. But when combined with the HTTP server (see above in section HTTP Server), the messages received by the SMTP server can be reproduced in JSON format.
When both the SMTP server and the HTTP server are enabled, the following additional endpoints are made available on the HTTP server:
/smtp/gui
A very basic HTML/JavaScript GUI that displays and auto-refreshes the received
messages is made available on the /smtp/gui
endpoint.
/smtp
On the /smtp
endpoint, an index of all received messages will be made
available as JSON in the following schema:
{
"count": 3,
"messages": [
{
"from": [
"testsender@programmfabrik.de"
],
"idx": 0,
"isMultipart": false,
"receivedAt": "2024-07-02T11:23:31.212023129+02:00",
"smtpFrom": "testsender@programmfabrik.de",
"smtpRcptTo": [
"testreceiver@programmfabrik.de"
],
"to": [
"testreceiver@programmfabrik.de"
]
},
{
"from": [
"testsender2@programmfabrik.de"
],
"idx": 1,
"isMultipart": true,
"receivedAt": "2024-07-02T11:23:31.212523916+02:00",
"smtpFrom": "testsender2@programmfabrik.de",
"smtpRcptTo": [
"testreceiver2@programmfabrik.de"
],
"subject": "Example Message",
"to": [
"testreceiver2@programmfabrik.de"
]
},
{
"from": [
"testsender3@programmfabrik.de"
],
"idx": 2,
"isMultipart": false,
"receivedAt": "2024-07-02T11:23:31.212773829+02:00",
"smtpFrom": "testsender3@programmfabrik.de",
"smtpRcptTo": [
"testreceiver3@programmfabrik.de"
],
"to": [
"testreceiver3@programmfabrik.de"
]
}
]
}
You can filter messages by passing one of more query parameters header
. header
can either be a JSON array of strings, or just a string. The filter checks that all headers (regexp format) match headers of the filtered email.
Headers that were encoded according to RFC2047 are decoded first.
/smtp/$idx
On the /smtp/$idx
endpoint (e.g. /smtp/1
), metadata about the message with the corresponding index is made available as JSON:
{
"bodySize": 306,
"contentType": "multipart/mixed",
"contentTypeParams": {
"boundary": "d36c3118be4745f9a1cb4556d11fe92d"
},
"from": [
"testsender2@programmfabrik.de"
],
"headers": {
"Content-Type": [
"multipart/mixed; boundary=\"d36c3118be4745f9a1cb4556d11fe92d\""
],
"Date": [
"Tue, 25 Jun 2024 11:15:57 +0200"
],
"From": [
"testsender2@programmfabrik.de"
],
"Mime-Version": [
"1.0"
],
"Subject": [
"Example Message"
],
"To": [
"testreceiver2@programmfabrik.de"
]
},
"idx": 1,
"isMultipart": true,
"multiparts": [
{
"bodySize": 15,
"contentType": "text/plain",
"contentTypeParams": {
"charset": "utf-8"
},
"headers": {
"Content-Type": [
"text/plain; charset=utf-8"
]
},
"idx": 0,
"isMultipart": false
},
{
"bodySize": 39,
"contentType": "text/html",
"contentTypeParams": {
"charset": "utf-8"
},
"headers": {
"Content-Type": [
"text/html; charset=utf-8"
]
},
"idx": 1,
"isMultipart": false
}
],
"multipartsCount": 2,
"receivedAt": "2024-07-02T12:54:44.443488367+02:00",
"smtpFrom": "testsender2@programmfabrik.de",
"smtpRcptTo": [
"testreceiver2@programmfabrik.de"
],
"subject": "Example Message",
"to": [
"testreceiver2@programmfabrik.de"
]
}
Headers that were encoded according to RFC2047 are decoded first.
/smtp/$idx/body
On the /smtp/$idx/body
endpoint (e.g. /smtp/1/body
), the message body (excluding message headers, including multipart part headers) is made availabe for the message with the corresponding index.
If the message was sent with a Content-Transfer-Encoding
of either base64
or quoted-printable
, the endpoint returns the decoded body.
If the message was sent with a Content-Type
header, it will be passed through to the HTTP response.
/smtp/$idx/multipart
For multipart messages, the /smtp/$idx/multipart
endpoint (e.g. /smtp/1/multipart
) will contain an index of that messages multiparts in the following schema:
{
"multiparts": [
{
"bodySize": 15,
"contentType": "text/plain",
"contentTypeParams": {
"charset": "utf-8"
},
"headers": {
"Content-Type": [
"text/plain; charset=utf-8"
]
},
"idx": 0,
"isMultipart": false
},
{
"bodySize": 39,
"contentType": "text/html",
"contentTypeParams": {
"charset": "utf-8"
},
"headers": {
"Content-Type": [
"text/html; charset=utf-8"
]
},
"idx": 1,
"isMultipart": false
}
],
"multipartsCount": 2
}
/smtp/$idx[/multipart/$partIdx]+
On the /smtp/$idx/multipart/$partIdx
endpoint (e.g. /smtp/1/multipart/0
), metadata about the multipart with the corresponding index is made available:
{
"bodySize": 15,
"contentType": "text/plain",
"contentTypeParams": {
"charset": "utf-8"
},
"headers": {
"Content-Type": [
"text/plain; charset=utf-8"
]
},
"idx": 0,
"isMultipart": false
}
Headers that were encoded according to RFC2047 are decoded first.
The endpoint can be called recursively for nested multipart messages, e.g. /smtp/1/multipart/0/multipart/1
.
/smtp/$idx[/multipart/$partIdx]+/body
On the /smtp/$idx/multipart/$partIdx/body
endpoint (e.g. /smtp/1/multipart/0/body
), the body of the multipart (excluding headers) is made available.
If the multipart was sent with a Content-Transfer-Encoding
of either base64
or quoted-printable
, the endpoint returns the decoded body.
If the message was sent with a Content-Type
header, it will be passed through to the HTTP response.
The endpoint can be called recursively for nested multipart messages, e.g. /smtp/1/multipart/0/multipart/1/body
.