hurl/integration
2023-07-15 13:38:35 +00:00
..
ssl Add isDate to existing integration tests 2023-07-11 13:31:58 +02:00
tests_error_lint Newline added at the end of the files 2022-10-24 15:04:50 +02:00
tests_error_parser Make URLs strictier: must begins with http://, https:// or {{. 2023-07-05 18:28:32 +00:00
tests_failed Fix URL runtime evaluation. 2023-07-06 13:04:17 +02:00
tests_ok Add connect-to per request option. 2023-07-15 13:38:35 +00:00
tests_ok_not_linted Use --data-binary for curl command when posting file. 2023-06-16 17:33:47 +02:00
app.py Split integration tests between tests_ok / tests_failed repository. 2022-02-14 16:24:39 +01:00
hurlfmt_check.sh hurlfmt Hurl files in tests_ok 2023-06-01 17:27:01 +02:00
integration.py hurlfmt Hurl files in tests_ok 2023-06-01 17:27:01 +02:00
README.md Add README explaining integration tests. 2023-06-21 15:54:13 +02:00
server.py Clean ci integration bindings 2022-09-19 16:05:52 +02:00
test_curl_commands.sh Fix false negatives on dockerised tests 2022-12-15 18:09:44 +01:00
test_echo.py Format python files with black formater 2022-02-05 18:15:41 +01:00
test_format.py Add options --in and --out in hurlfmt 2023-04-18 08:34:44 +02:00
test_html_output.py Newline added at the end of the files 2022-10-24 15:04:50 +02:00
test_lint.py Format python files with black formater 2022-02-05 18:15:41 +01:00
test_script.py Print actual/expected lines when number of lines does not match 2023-06-01 17:17:35 +02:00

Hurl Integration Tests Suite

Introduction

In the Hurl project, there are three type of tests:

  • Rust unit tests: run at the root of the project with cargo test --lib
  • Rust integration tests: run at the root of the project with cargo test. You will also run unit test with this command. To run Rust integration tests, you need to launch a local test server (see below).

These tests are "classic" Rust tests and should not surprise a Rust developer.

Along with tests, we have an extensive integration test suite. These tests launch scripts to run hurl and hurlfmt and test various options and Hurl files. To run these tests you have to set up a local server (see below) All these tests will be performed automatically in Hurl CI/CD, on various OS, for every pull request.

Set up Test Local Server

Python 3.9+

The local test server is a Flask application, so you will need Python 3.9+ on your machine. You can create a Python virtual environment and install required dependencies:

$ python3 -m venv .venv
$ source .venv/bin/activate
$ pip install --requirements bin/requirements-fozen.txt

Proxy

Some integration tests need a proxy. You can use mitmproxy or squid.

Start local server

You can use the scripts bin/test/test_prerequites.sh / bin/test/test_prerequites.ps1 depending on your OS to start the local test server and proxy. Once launch, there is:

Now, everything is ready to run the integration tests!

Integration Tests

Organisation

Integration tests are under integration directory :

  • tests_ok: every test there must be successful (exit code 0). The Hurl files in this folder are formatted with hurlfmt.
  • tests_ok_not_linted: every test here must be successful but are not necessary formatted with hurlfmt. This way we can ensure that there is no regression even if a Hurl file doesn't follow a stricter format.
  • tests_failed: every test must fail (exit code different from 0). Tests are syntactically correct, so the error raised by the test is a runtime error.
  • tests_error_parser: every test is not a syntactically correct Hurl file. We test here the parsing error message.
  • tests_error_lint: every test is syntactically correct, but is not formatted through hurlfmt. We test here the linting error message.

Files Description

An integration test consists of:

  • two runnable scripts: one for Linux, macOS (foo.sh) and one for Windows (foo.ps1). These are the integration tests that we want to execute.
  • a Hurl file (foo.hurl)
  • a Flask endpoint (foo.py). This is the server side used by the Hurl file. You can add as many assert as you want to test that our Hurl client conforms to what is expected. Generally, each integration test has its own Flask endpoint, even if there is some duplication between tests.
  • an expected stdout file (foo.out). This file is the expected value for stdout. This file is not dependant from the OS, as we want a Hurl file to have the same stdout on any OS. If the stdout have some variant datas (like timestamp for instance), one can use a patterned expected file, with ~~~ for wildcard matching (foo.out.pattern)
  • an expected stderr file (foo.err). This file is the expected value for stderr. This file is not dependant from the OS, as we want a Hurl file to have the same stderr on any OS. Like stdout, stderr expected file can be patterned (foo.err.pattern)
  • an expected exit code (foo.exit). This file is the expected value of the script. If absent, the default exit code is 0.
  • an expected HTML export of the Hurl source file (foo.html)
  • an expected JSON export of the Hurl source file (foo.json). Note: this is not the stdout output of Hurl using --json. This is a JSON view of the Hurl source file and can serve to convert from/to Hurl format.
  • an expected list of curl commands. This list is the curl command equivalent to each request in the Hurl file (which is logged in --verbose/--very-verbose mode). Each curl command is run against the server.

To run all integration tests:

$ cd integration
$ python3 integration.py

To run a particular integration test without any check:

$ cd integration
$ tests_ok/hello.sh

To run a particular integration test with all check (stdout, stderr, HTML/JSON export etc...):

$ cd integration
$ python3 test_script.py tests_ok/hello.sh

Sample

include.sh:

#!/bin/bash
set -Eeuo pipefail
hurl tests_ok/include.hurl --include --verbose

include.ps1:

Set-StrictMode -Version latest
$ErrorActionPreference = 'Stop'
hurl tests_ok/include.hurl --include --verbose

include.hurl:

GET http://localhost:8000/include

HTTP 200
`Hello`

include.py:

from app import app
from flask import Response


@app.route("/include")
def include():
    return Response("Hello")

include.out.pattern:

HTTP/1.1 200
Server: Werkzeug/~~~ Python/~~~
Date: ~~~
Content-Type: text/html; charset=utf-8
Content-Length: 5
Server: Flask Server
Connection: close

Hello

include.html:

<pre><code class="language-hurl"><span class="hurl-entry"><span class="request"><span class="line"><span class="method">GET</span> <span class="url">http://localhost:8000/include</span></span>
</span><span class="response"><span class="line"></span>
<span class="line"><span class="version">HTTP</span> <span class="number">200</span></span>
<span class="line"><span class="string">`Hello`</span></span>
</span></span><span class="line"></span>
</code></pre>

include.json:

{"entries":[{"request":{"method":"GET","url":"http://localhost:8000/include"},"response":{"status":200,"body":{"type":"text","value":"Hello"}}}]}

include.curl:

curl 'http://localhost:8000/include'