Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/michaelo/sapt
Simple file-oriented API-testing tool
https://github.com/michaelo/sapt
api cli curl testing zig
Last synced: 2 months ago
JSON representation
Simple file-oriented API-testing tool
- Host: GitHub
- URL: https://github.com/michaelo/sapt
- Owner: michaelo
- License: mit
- Created: 2021-09-15T23:01:03.000Z (over 3 years ago)
- Default Branch: main
- Last Pushed: 2024-02-01T15:46:05.000Z (12 months ago)
- Last Synced: 2024-08-04T04:04:58.279Z (6 months ago)
- Topics: api, cli, curl, testing, zig
- Language: Zig
- Homepage:
- Size: 1.36 MB
- Stars: 12
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-zig - sapt🗒️Simple file-oriented API-testing tool
README
sapt - A simple tool for API testing
==============
*Att: I'm testing different intro-texts to test what communicates the intention the best*sapt aims to be a simple tool to help with API-testing and similar use cases. It focuses on making it easy for developers to compose, organize and perform tests/requests in an open, reliable and source-control friendly way.
sapt *"is not"* a full-fledged GUI-based do-anything tool, but rather a focused command line utility.
Usage: Basic
-------------sapt requires you to organise your requests in individual files. Those files may be gathered in folders to create test suites.
testsuite/01-mytest.pi contents:
> GET https://api.warnme.no/api/status
< 200Running a single test:
% sapt testsuite/01-mytest.pi
1/1 testsuite/01-mytest.pi : OKHelp:
% sapt -h
sapt v1.0.0 - Simple API TesterUsage: sapt [arguments] [file1.pi file2.pi ... fileN.pi]
Examples:
sapt api_is_healthy.pi
sapt testsuite01/
sapt -b=myplaybook.book
sapt -i=generaldefs/.env testsuite01/Arguments:
--colors=auto|on|off Set wether to attempt to use colored output or not
--delay=NN Delay execution of each consecutive step with NN ms
-e, --early-quit Abort upon first non-successful test
-h, --help Show this help and exit
--help-format Show details regarding file formats and exit
-i=file,
--initial-vars=file Provide file with variable-definitions made available
to all tests
--insecure Don't verify SSL certificates
-m, --multithread Activates multithreading - relevant for repeated
tests via playbooks
-p, --pretty Try to format response data based on Content-Type.
Naive support for JSON, XML and HTML
-b=file,
--playbook=file Read tests to perform from playbook-file -- if set,
ignores other tests passed as arguments
-d, --show-response Show response data. Even if -s.
-s, --silent Silent. Suppresses output. Overrules verbose.
-v, --verbose Verbose output
--verbose-curl Verbose output from libcurl
--version Show version and exit-DKEY=VALUE Define variable, similar to .env-files. Can be set
multiple timessapt can take multiple arguments, both files and folders. The entire input-set will be sorted alphanumerically when passing a folder, thus you can dictate the order of execution by making sure the names of the scripts reflects the order:
* suite/01-auth.pi
* suite/02-post-entry.pi
* suite/03-get-posted-entry.pi
* suite/04-delete-posted-entry.pi*Note: playbooks provides a way to override this.*
Usage: Complex
----------------Assuming you first have to get an authorization code from an auth-endpoint, which you will need in other tests.
Let's say you have the following files:
* myservice/.env
* myservice/01-auth.pi
* myservice/02-get-data.pi### Set up variables: myservice/.env
OIDC_USERNAME=myoidcclient
OIDC_PASSWORD=sup3rs3cr3t
USERNAME=someuser
PASSWORD=supersecret42### Get the auth-token: myservice/01-auth.pi
> POST https://my.service/api/auth
Authorization: Basic {{base64enc({{OIDC_USERNAME}}:{{OIDC_PASSWORD}})}}
Content-Type: application/x-www-form-urlencoded
-
grant_type=password&username={{USERNAME}}&password={{PASSWORD}}&scope=openid%20profile
< 200
id_token="id_token":"()"Provided that the auth-endpoint will return something like this:
{"access_token": "...", "id_token":"...", "...", ...}
... the test will then set the id_token-variable, allowing it to be referred in subsequent tests.
### Get data from service using data from previous test: myservice/02-get-data.pi
> GET https://my.app/api/entry
Cookie: SecurityToken={{id_token}}
< 200### Finally, Run the testsuite
sapt myservice
Output:
1: myservice/01-auth.pi :OK (HTTP 200)
2: myservice/02-get-data.pi :OK (HTTP 200)
------------------
2/2 OK
------------------
FINISHED - total time: 0.189s*Tips: You can add -v or -d for more detailed output*
Usage: playbook
-----------
myplay.book contents:# Run this request 1 time
myproj/auth.pi
# Run this request 100 times
myproj/api_get.pi * 100Tests shall be run in the order declared in the playbook. Each test may be followed by a number indicating the number of times it shall be performed.
Running the playbook:
sapt -b=myplay.book
Playbooks resolves paths relative to its own location.
Output:
1/5: myproj/auth.pi : OK (HTTP 200 - OK)
time: 256ms
2/5: myproj/api_get.pi : OK (HTTP 200 - OK)
100 iterations. 100 OK, 0 Error
time: 1050ms/100 iterations [83ms-215ms] avg:105ms
------------------
2/2 OK
------------------Build:
------------
The tool is written in [zig](https://ziglang.org/) v0.9.0, and depends on [libcurl](https://curl.se/libcurl/).Prerequisites:
* [zig is installed](https://ziglang.org/download/) and available in path. Development is done on 0.9.0
* [libcurl is installed](https://curl.se/download.html) and library and headers are available in either path or through pkg-config.Get source:
git clone https://github.com/michaelo/sapt
cd saptDevelopment build/run:
zig build run
Run all tests:
zig build test
Install:
zig build install --prefix-exe-dir /usr/local/bin
*... or other path to put the executable to be in path.*
Design goals:
------------
* Only you should own and control your data - e.g. any version control and data sharing is up to you.
* Tests should be easily written and arranged - no hidden headers or such
* Support easy use of secrets and common variables/definitions
* Tests should live alongside the artifacts they testsTerminology:
------
Term
Description
test
the particular file/request to be processed. A test can result in "OK" or "ERROR"
playbook
a particular recipe of tests, their order and other parameters to be executed in a particular fashion
extraction-expression
The mechanism which allows one to extract data from responses according to a given expression and store the results in a variable for use in following tests. E.g. extract an auth-token to use in protected requests.
Limitations:
------
Due in part to the efforts to both having a clear understanding of the RAM-usage, as well as keeping the heap-usage low and controlled, a set of discrete limitations are currently characteristic for the tool. I will very likely revise a lot of these decisions going forward - but here they are:
What
Limitation
General limitations
Max number of tests to process in a given run
128
Max length for any file path
1024
Max repeats for single test
1000
Test-specific parameters
Max number of headers
32
Max length for a given HTTP-header
8K
Max number of variables+functions in a given test
64
Max length of a function response
1024
Max length of a variable key
128
Max length of a variable value
8K
Max URL length
2048
Max size of payload
1M
Extraction-expressions: order of precedence
sapt will first attempt to match extraction entries with response body first, response headers second.
Test-file specification:
--------
Comments:
# Comment - must be start of line. Can be used everywhere except of in payload-section
Variables:
{{my_var}}
Functions:
Convenience-functions are (to be) implemented. The argument can consist of other variables.
{{base64enc(string)}}
Example:
Authorization: basic {{base64enc({{username}}:{{password}})}}
Supported functions:
* base64enc(string) - base64-encoding
* env(string) - lookup variables from OS-environment
* *TODO: urlencode(string)*
* *TODO: base64dec(string) ?*Input section:
# '>' marks start of 'input'-section, followed by HTTP verb and URL
> POST https://some.where/
# List of HTTP-headers, optional
Content-Type: application/json
Accept: application/jsonPayload section, optional:
# '-' marks start of payload-section, and it goes until output-section
-
{"some":"data}*TBD: Implement support for injecting files? If so: allow arbitrary sizes.*
Output section:
# [optional string to check response for it to be considered successful]
< 200 optional text# HTTP-code '0' is "don't care"
< 0Set of variable extraction expressions, optional:
# Key=expression
# expression format:
# Expression shall consists of a string representing the output, with '()' indicating the part to extract
AUTH_TOKEN="token":"()"*TBD: Might support more regex-like expressions to control e.g. character groups and such.*
Exit codes
-------------
sapt is also usable in e.g. cron jobs to monitor availability of a service.This is a tentative list of exit codes currently implemented in sapt:
* 0: OK
* 1: Something went wrong processing the input set
* 2: One or more tests failedUse cases
-------------
This sections aims to provide a set of example use cases for which sapt can be useful. This is not an exchaustive list, but please let me know if any other use cases are found:* Test / explore the behaviour of an API / web service
* Describe and verify desired behviour when developing a service - e.g. TDD-like process
* Monitor a set of relevant endpoints to quickly debug which - if any - one fails
* Cron-job to monitor a service
* Load-testingTODO, somewhat ordered:
------------
*Att! The points here are not changes that we strictly commit to, but an organic list of things to consider. Higher up = more likely** Separate automagic handling between .env-file explicitly and implicitly passed (through folder)? Explicit should perhaps ignore folder-rules?
* Determine if current solution where variables can't be overwritten is a good idea or not.
* Libs/deps handling:
* Credits: Determine all deps we need to ship.
* libcurl for all platforms - license added to CREDITS
* zlib for Windows - license added to CREDITS
* OpenSSL? Other curl-deps?
* Look into staticly linking all deps - the absolute best would be a single, self-contained executable
* Get proper version-control of which dynamic libraries we depend on/provide.
* Dev: Set up automatic builds/cross-builds for Win10 x64, Linux x64, macOS (x64 and Arm)
* Clean up lib-handling. Currently we e.g. have libcurl stored as libcurl.dll and curl.dll due to some linkage-discrepencies for Windows/vcpkg. Can we simply vendor it?
* Append current git hash (short) for debug-builds. Need to figure out how to detect this at compile-time and inject it properly.
* Due to this being an explorative project while learning Zig, there are inconsistencies regarding memory-handling. This must be cleaned up and verified.
* Code quality - especially in main.zig - is quite crap at this point.
* More advanced sequence options?:
* Specify setup/teardown-tests to be run before/after test/suite/playbook?
* --suite-setup, --suite-teardown, --step-setup, --step-teardown?
* What about playbooks?
* Specify option to only re-run specific steps of a sequence? --steps=1,3,5-7.
* Provide better stats for repeats. We currently have min, max and avg/mean time. Could median or something mode-like be as useful or more? A plot would be nice here.
* Also ensure that total time for entire test set also prints the accumulated value of each step without the delay set by --delay
* Describe/explore how/if --delay shall affect playbooks. Currently: it doesn't. Playbooks should be self-contained, so we'd need an in-playbook variant of the functionality
* Implement support to do step-by-step tests by e.g. requiring user to press enter between each test?
* Store responses? E.g. 'sapt mysuite/ --store-responses=./out/' creates ./out/mysuite/01-test1.pi.out etc
* Playbooks:
* TBD: What shall the semantics be regarding response data and variable-extraction when we have multiple repetitions? Makes no sense perhaps, so either have "last result matters", "undefined behaviour" or "unsupported". Wait for proper use cases.
* Test/verify safety of string lengths: parsing
* Support both keeping variables between (default) as well as explicitly allowing sandboxing (flag) of tests?
* TBD: Shall we support "repeats" in test-files as well, not only in playbooks?
* Actively limit the set of protocols we allow. We currently just forward the URL however it is to CURL. If we staticly link, build a custom variant with only the feature set we need.
* Finish basic syntax highligh ruleset for the test-files
* Dev: Test feature flags based on comptime-parsing a feature-fileFeature-exploration AKA Maybe-TODO:
-------------
* Support handling encrypted variables?
* Support list of curl-commands as alternative output?
* Performant, light-weight GUI (optional)? Plotting performance for stress tests and such.
* Support response-time as test-prereq? Perhaps in playlist (low pri)
* TBD: Possibility to set "verbose" only for a specific test? Using e.g. the test-sequence-number?
* TBD: Actually parse response data - which also can allow us to more precisely extract data semanticallyThanks / attributions:
--------
* zig - an interesting language of which this project is my first deliverable
* libcurl - the workhorse*See ATTRIBUTIONS.md for licenses.*