mirror of
https://github.com/httpie/cli.git
synced 2025-01-21 12:58:37 +01:00
4f1c9441c5
* Fix encoding error with non-prettified encoded responses
Removed `--format-option response.as` an promote `--response-as`: using
the format option would be misleading as it is now also used by non-prettified
responses.
* Encoding refactoring
* split --response-as into --response-mime and --response-charset
* add support for Content-Type charset for requests printed to terminal
* add support charset detection for requests printed to terminal without a Content-Type charset
* etc.
* `test_unicode.py` → `test_encoding.py`
* Drop sequence length check
* Clean-up tests
* [skip ci] Tweaks
* Use the compatible release clause for `charset_normalizer` requirement
Cf. https://www.python.org/dev/peps/pep-0440/#version-specifiers
* Clean-up
* Partially revert d52a4833e4
* Changelog
* Tweak tests
* [skip ci] Better test name
* Cleanup tests and add request body charset detection
* More test suite cleanups
* Cleanup
* Fix code style in test
* Improve detect_encoding() docstring
* Uniformize pytest.mark.parametrize() calls
* [skip ci] Comment out TODOs (will be tackled in a specific PR)
Co-authored-by: Jakub Roztocil <jakub@roztocil.co>
218 lines
5.8 KiB
Python
218 lines
5.8 KiB
Python
"""High-level tests."""
|
|
import io
|
|
from unittest import mock
|
|
|
|
import pytest
|
|
|
|
import httpie
|
|
import httpie.__main__
|
|
from .fixtures import FILE_CONTENT, FILE_PATH
|
|
from httpie.cli.exceptions import ParseError
|
|
from httpie.context import Environment
|
|
from httpie.encoding import UTF8
|
|
from httpie.status import ExitStatus
|
|
from .utils import HTTP_OK, MockEnvironment, StdinBytesIO, http
|
|
|
|
|
|
def test_main_entry_point():
|
|
# Patch stdin to bypass pytest capture
|
|
with mock.patch.object(Environment, 'stdin', io.StringIO()):
|
|
assert httpie.__main__.main() == ExitStatus.ERROR.value
|
|
|
|
|
|
@mock.patch('httpie.core.main')
|
|
def test_main_entry_point_keyboard_interrupt(main):
|
|
main.side_effect = KeyboardInterrupt()
|
|
with mock.patch.object(Environment, 'stdin', io.StringIO()):
|
|
assert httpie.__main__.main() == ExitStatus.ERROR_CTRL_C.value
|
|
|
|
|
|
def test_debug():
|
|
r = http('--debug')
|
|
assert r.exit_status == ExitStatus.SUCCESS
|
|
assert f'HTTPie {httpie.__version__}' in r.stderr
|
|
|
|
|
|
def test_help():
|
|
r = http('--help', tolerate_error_exit_status=True)
|
|
assert r.exit_status == ExitStatus.SUCCESS
|
|
assert 'https://github.com/httpie/httpie/issues' in r
|
|
|
|
|
|
def test_version():
|
|
r = http('--version', tolerate_error_exit_status=True)
|
|
assert r.exit_status == ExitStatus.SUCCESS
|
|
assert httpie.__version__ == r.strip()
|
|
|
|
|
|
def test_GET(httpbin_both):
|
|
r = http('GET', httpbin_both + '/get')
|
|
assert HTTP_OK in r
|
|
|
|
|
|
def test_path_dot_normalization():
|
|
r = http(
|
|
'--offline',
|
|
'example.org/../../etc/password',
|
|
'param==value'
|
|
)
|
|
assert 'GET /etc/password?param=value' in r
|
|
|
|
|
|
def test_path_as_is():
|
|
r = http(
|
|
'--offline',
|
|
'--path-as-is',
|
|
'example.org/../../etc/password',
|
|
'param==value'
|
|
)
|
|
assert 'GET /../../etc/password?param=value' in r
|
|
|
|
|
|
def test_DELETE(httpbin_both):
|
|
r = http('DELETE', httpbin_both + '/delete')
|
|
assert HTTP_OK in r
|
|
|
|
|
|
def test_PUT(httpbin_both):
|
|
r = http('PUT', httpbin_both + '/put', 'foo=bar')
|
|
assert HTTP_OK in r
|
|
assert r.json['json']['foo'] == 'bar'
|
|
|
|
|
|
def test_POST_JSON_data(httpbin_both):
|
|
r = http('POST', httpbin_both + '/post', 'foo=bar')
|
|
assert HTTP_OK in r
|
|
assert r.json['json']['foo'] == 'bar'
|
|
|
|
|
|
def test_POST_form(httpbin_both):
|
|
r = http('--form', 'POST', httpbin_both + '/post', 'foo=bar')
|
|
assert HTTP_OK in r
|
|
assert '"foo": "bar"' in r
|
|
|
|
|
|
def test_POST_form_multiple_values(httpbin_both):
|
|
r = http('--form', 'POST', httpbin_both + '/post', 'foo=bar', 'foo=baz')
|
|
assert HTTP_OK in r
|
|
assert r.json['form'] == {
|
|
'foo': ['bar', 'baz']
|
|
}
|
|
|
|
|
|
def test_POST_raw(httpbin_both):
|
|
r = http('--raw', 'foo bar', 'POST', httpbin_both + '/post')
|
|
assert HTTP_OK in r
|
|
assert '"foo bar"' in r
|
|
|
|
|
|
def test_POST_stdin(httpbin_both):
|
|
env = MockEnvironment(
|
|
stdin=StdinBytesIO(FILE_PATH.read_bytes()),
|
|
stdin_isatty=False,
|
|
)
|
|
r = http('--form', 'POST', httpbin_both + '/post', env=env)
|
|
assert HTTP_OK in r
|
|
assert FILE_CONTENT in r
|
|
|
|
|
|
def test_POST_file(httpbin_both):
|
|
r = http('--form', 'POST', httpbin_both + '/post', f'file@{FILE_PATH}')
|
|
assert HTTP_OK in r
|
|
assert FILE_CONTENT in r
|
|
|
|
|
|
def test_form_POST_file_redirected_stdin(httpbin):
|
|
"""
|
|
<https://github.com/httpie/httpie/issues/840>
|
|
|
|
"""
|
|
with open(FILE_PATH, encoding=UTF8):
|
|
r = http(
|
|
'--form',
|
|
'POST',
|
|
httpbin + '/post',
|
|
f'file@{FILE_PATH}',
|
|
tolerate_error_exit_status=True,
|
|
env=MockEnvironment(
|
|
stdin=StdinBytesIO(FILE_PATH.read_bytes()),
|
|
stdin_isatty=False,
|
|
),
|
|
)
|
|
assert r.exit_status == ExitStatus.ERROR
|
|
assert 'cannot be mixed' in r.stderr
|
|
|
|
|
|
def test_raw_POST_key_values_supplied(httpbin):
|
|
r = http(
|
|
'--raw',
|
|
'foo bar',
|
|
'POST',
|
|
httpbin + '/post',
|
|
'foo=bar',
|
|
tolerate_error_exit_status=True,
|
|
)
|
|
assert r.exit_status == ExitStatus.ERROR
|
|
assert 'cannot be mixed' in r.stderr
|
|
|
|
|
|
def test_raw_POST_redirected_stdin(httpbin):
|
|
r = http(
|
|
'--raw',
|
|
'foo bar',
|
|
'POST',
|
|
httpbin + '/post',
|
|
tolerate_error_exit_status=True,
|
|
env=MockEnvironment(
|
|
stdin='some=value',
|
|
stdin_isatty=False,
|
|
),
|
|
)
|
|
assert r.exit_status == ExitStatus.ERROR
|
|
assert 'cannot be mixed' in r.stderr
|
|
|
|
|
|
def test_headers(httpbin_both):
|
|
r = http('GET', httpbin_both + '/headers', 'Foo:bar')
|
|
assert HTTP_OK in r
|
|
assert '"User-Agent": "HTTPie' in r, r
|
|
assert '"Foo": "bar"' in r
|
|
|
|
|
|
def test_headers_unset(httpbin_both):
|
|
r = http('GET', httpbin_both + '/headers')
|
|
assert 'Accept' in r.json['headers'] # default Accept present
|
|
|
|
r = http('GET', httpbin_both + '/headers', 'Accept:')
|
|
assert 'Accept' not in r.json['headers'] # default Accept unset
|
|
|
|
|
|
@pytest.mark.skip('unimplemented')
|
|
def test_unset_host_header(httpbin_both):
|
|
r = http('GET', httpbin_both + '/headers')
|
|
assert 'Host' in r.json['headers'] # default Host present
|
|
|
|
r = http('GET', httpbin_both + '/headers', 'Host:')
|
|
assert 'Host' not in r.json['headers'] # default Host unset
|
|
|
|
|
|
def test_headers_empty_value(httpbin_both):
|
|
r = http('GET', httpbin_both + '/headers')
|
|
assert r.json['headers']['Accept'] # default Accept has value
|
|
|
|
r = http('GET', httpbin_both + '/headers', 'Accept;')
|
|
assert r.json['headers']['Accept'] == '' # Accept has no value
|
|
|
|
|
|
def test_headers_empty_value_with_value_gives_error(httpbin):
|
|
with pytest.raises(ParseError):
|
|
http('GET', httpbin + '/headers', 'Accept;SYNTAX_ERROR')
|
|
|
|
|
|
def test_json_input_preserve_order(httpbin_both):
|
|
r = http('PATCH', httpbin_both + '/patch',
|
|
'order:={"map":{"1":"first","2":"second"}}')
|
|
assert HTTP_OK in r
|
|
assert r.json['data'] == \
|
|
'{"order": {"map": {"1": "first", "2": "second"}}}'
|