Merge remote-tracking branch 'upstream/master'

This commit is contained in:
Michael Floering 2016-01-22 18:40:53 -06:00
commit 018e1f68de
21 changed files with 151 additions and 85 deletions

View File

@ -33,3 +33,4 @@ Patches and ideas
* `Matthias Lehmann <https://github.com/matleh>`_ * `Matthias Lehmann <https://github.com/matleh>`_
* `Dennis Brakhane <https://github.com/brakhane>`_ * `Dennis Brakhane <https://github.com/brakhane>`_
* `Matt Layman <https://github.com/mblayman>`_ * `Matt Layman <https://github.com/mblayman>`_
* `Edward Yang <https://github.com/honorabrutroll>`_

View File

@ -5,13 +5,20 @@ Change Log
This document records all notable changes to `HTTPie <http://httpie.org>`_. This document records all notable changes to `HTTPie <http://httpie.org>`_.
This project adheres to `Semantic Versioning <http://semver.org/>`_. This project adheres to `Semantic Versioning <http://semver.org/>`_.
`1.0.0-dev`_ (Unreleased) `1.0.0-dev`_ (Unreleased)
------------------------- -------------------------
`0.9.3`_ (2016-01-01)
-------------------------
* Changed the default color ``--style`` from ``solarized`` to ``monokai`` * Changed the default color ``--style`` from ``solarized`` to ``monokai``
* Added Bash auto complete support * Added basic Bash autocomplete support (need to be installed manually)
* Added request details to connection error messages * Added request details to connection error messages
* Fixed ``'requests.packages.urllib3' has no attribute 'disable_warnings'``
errors that occurred in some installations
* Fixed colors and formatting on Windows
* Fixed ``--auth`` prompt on Windows
`0.9.2`_ (2015-02-24) `0.9.2`_ (2015-02-24)
@ -252,4 +259,5 @@ This project adheres to `Semantic Versioning <http://semver.org/>`_.
.. _0.9.0: https://github.com/jkbrzt/httpie/compare/0.8.0...0.9.0 .. _0.9.0: https://github.com/jkbrzt/httpie/compare/0.8.0...0.9.0
.. _0.9.1: https://github.com/jkbrzt/httpie/compare/0.9.0...0.9.1 .. _0.9.1: https://github.com/jkbrzt/httpie/compare/0.9.0...0.9.1
.. _0.9.2: https://github.com/jkbrzt/httpie/compare/0.9.1...0.9.2 .. _0.9.2: https://github.com/jkbrzt/httpie/compare/0.9.1...0.9.2
.. _1.0.0-dev: https://github.com/jkbrzt/httpie/compare/0.9.2...master .. _0.9.3: https://github.com/jkbrzt/httpie/compare/0.9.2...0.9.3
.. _1.0.0-dev: https://github.com/jkbrzt/httpie/compare/0.9.3...master

View File

@ -1,12 +1,13 @@
######################
Contributing to HTTPie Contributing to HTTPie
###################### ######################
Bug reports and code and documentation patches are greatly appretiated. You can Bug reports and code and documentation patches are welcome. You can
also help by using the development version of HTTPie and reporting any bugs you help this project also by using the development version of HTTPie
might encounter. and by reporting any bugs you might encounter.
Bug Reports 1. Reporting bugs
=========== =================
**It's important that you provide the full command argument list **It's important that you provide the full command argument list
as well as the output of the failing command.** as well as the output of the failing command.**
@ -15,12 +16,12 @@ to your bug report, e.g.:
.. code-block:: bash .. code-block:: bash
$ http --debug [arguments that trigger the error] $ http --debug [COMPLETE ARGUMENT LIST THAT TRIGGERS THE ERROR]
[complete output] [COMPLETE OUTPUT]
Contributing Code and Documentation 2. Contributing Code and Docs
=================================== =============================
Before working on a new feature or a bug, please browse `existing issues`_ Before working on a new feature or a bug, please browse `existing issues`_
to see whether it has been previously discussed. If the change in question to see whether it has been previously discussed. If the change in question
@ -28,8 +29,11 @@ is a bigger one, it's always good to discuss before your starting working on
it. it.
Development Environment Creating Development Environment
----------------------- --------------------------------
Go to https://github.com/jkbrzt/httpie and fork the project repository.
.. code-block:: bash .. code-block:: bash
@ -52,44 +56,61 @@ Making Changes
Please make sure your changes conform to `Style Guide for Python Code`_ (PEP8). Please make sure your changes conform to `Style Guide for Python Code`_ (PEP8).
Tests Testing
----- -------
Before opening a pull requests, please make sure the `test suite`_ passes Before opening a pull requests, please make sure the `test suite`_ passes
in all of the `supported Python environments`_. You should also **add tests in all of the `supported Python environments`_. You should also add tests
for any new features and bug fixes**. for any new features and bug fixes.
HTTPie uses `pytest`_ and `Tox`_. HTTPie uses `pytest`_ and `Tox`_ for testing.
Running all tests:
******************
.. code-block:: bash .. code-block:: bash
### Running all tests: # Run all tests on the current Python interpreter
# Current Python
make test make test
# Current Python with coverage # Run all tests on the current Python with coverage
make test-cover make test-cover
# All the supported and available Pythons via Tox # Run all tests in all of the supported and available Pythons via Tox
make test-tox make test-tox
### Running specific tests: # Run all tests for code as well as packaging, etc.
make test-all
# Current Python
pytest tests/test_uploads.py
# All Pythons Running specific tests:
***********************
.. code-block:: bash
# Run specific tests on the current Python
py.test tests/test_uploads.py
py.test tests/test_uploads.py::TestMultipartFormDataFileUpload
py.test tests/test_uploads.py::TestMultipartFormDataFileUpload::test_upload_ok
# Run specific tests on the on all Pythons via Tox
tox -- tests/test_uploads.py --verbose tox -- tests/test_uploads.py --verbose
tox -- tests/test_uploads.py::TestMultipartFormDataFileUpload --verbose
tox -- tests/test_uploads.py::TestMultipartFormDataFileUpload::test_upload_ok --verbose
Don't forget to add yourself to `AUTHORS.rst`_. -----
See `Makefile`_ for additional development utilities.
Don't forget to add yourself to `AUTHORS`_!
.. _Tox: http://tox.testrun.org .. _Tox: http://tox.testrun.org
.. _supported Python environments: https://github.com/jkbrzt/httpie/blob/master/tox.ini .. _supported Python environments: https://github.com/jkbrzt/httpie/blob/master/tox.ini
.. _existing issues: https://github.com/jkbrzt/httpie/issues?state=open .. _existing issues: https://github.com/jkbrzt/httpie/issues?state=open
.. _AUTHORS.rst: https://github.com/jkbrzt/httpie/blob/master/AUTHORS.rst .. _AUTHORS: https://github.com/jkbrzt/httpie/blob/master/AUTHORS.rst
.. _Makefile: https://github.com/jkbrzt/httpie/blob/master/Makefile
.. _pytest: http://pytest.org/ .. _pytest: http://pytest.org/
.. _Style Guide for Python Code: http://python.org/dev/peps/pep-0008/ .. _Style Guide for Python Code: http://python.org/dev/peps/pep-0008/
.. _test suite: https://github.com/jkbrzt/httpie/tree/master/tests .. _test suite: https://github.com/jkbrzt/httpie/tree/master/tests

View File

@ -1,4 +1,4 @@
Copyright © 2012-2015 Jakub Roztocil <jakub@roztocil.co> Copyright © 2012-2016 Jakub Roztocil <jakub@roztocil.co>
Redistribution and use in source and binary forms, with or without Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met: modification, are permitted provided that the following conditions are met:

View File

@ -1,39 +1,42 @@
#
# See ./CONTRIBUTING.rst
#
VERSION=$(shell grep __version__ httpie/__init__.py) VERSION=$(shell grep __version__ httpie/__init__.py)
REQUIREMENTS="requirements-dev.txt" REQUIREMENTS="requirements-dev.txt"
TAG="\n\n\033[0;32m\#\#\# " TAG="\n\n\033[0;32m\#\#\# "
END=" \#\#\# \033[0m\n" END=" \#\#\# \033[0m\n"
all: test all: test
uninstall-httpie:
@echo $(TAG)Removing existing installation of HTTPie$(END)
- pip uninstall --yes httpie >/dev/null
! which http
@echo
uninstall-all: uninstall-httpie
- pip uninstall --yes -r $(REQUIREMENTS)
init: uninstall-httpie init: uninstall-httpie
@echo $(TAG)Installing dev requirements$(END) @echo $(TAG)Installing dev requirements$(END)
pip install --upgrade -r $(REQUIREMENTS) pip install --upgrade -r $(REQUIREMENTS)
@echo $(TAG)Installing HTTPie$(END) @echo $(TAG)Installing HTTPie$(END)
pip install --upgrade --editable . pip install --upgrade --editable .
@echo @echo
test: init test: init
@echo $(TAG)Running tests in on current Python with coverage $(END) @echo $(TAG)Running tests on the current Python interpreter with coverage $(END)
py.test --cov ./httpie --cov ./tests --doctest-modules --verbose ./httpie ./tests py.test --cov ./httpie --cov ./tests --doctest-modules --verbose ./httpie ./tests
@echo @echo
test-tox: init test-tox: init
@echo $(TAG)Running tests on all Pythons via Tox$(END) @echo $(TAG)Running tests on all Pythons via Tox$(END)
tox tox
@echo @echo
test-dist: test-sdist test-bdist-wheel test-dist: test-sdist test-bdist-wheel
@echo @echo
test-sdist: clean uninstall-httpie test-sdist: clean uninstall-httpie
@echo $(TAG)Testing sdist build an installation$(END) @echo $(TAG)Testing sdist build an installation$(END)
python setup.py sdist python setup.py sdist
@ -41,6 +44,7 @@ test-sdist: clean uninstall-httpie
which http which http
@echo @echo
test-bdist-wheel: clean uninstall-httpie test-bdist-wheel: clean uninstall-httpie
@echo $(TAG)Testing wheel build an installation$(END) @echo $(TAG)Testing wheel build an installation$(END)
python setup.py bdist_wheel python setup.py bdist_wheel
@ -48,9 +52,11 @@ test-bdist-wheel: clean uninstall-httpie
which http which http
@echo @echo
# This tests everything, even this Makefile. # This tests everything, even this Makefile.
test-all: uninstall-all clean init test test-tox test-dist test-all: uninstall-all clean init test test-tox test-dist
publish: test-all publish: test-all
@echo $(TAG)Testing wheel build an installation$(END) @echo $(TAG)Testing wheel build an installation$(END)
@echo "$(VERSION)" @echo "$(VERSION)"
@ -60,8 +66,29 @@ publish: test-all
python setup.py bdist_wheel upload python setup.py bdist_wheel upload
@echo @echo
clean: clean:
@echo $(TAG)Cleaning up$(END) @echo $(TAG)Cleaning up$(END)
rm -rf .tox *.egg dist build .coverage rm -rf .tox *.egg dist build .coverage
find . -name '__pycache__' -delete -print -o -name '*.pyc' -delete -print find . -name '__pycache__' -delete -print -o -name '*.pyc' -delete -print
@echo @echo
uninstall-httpie:
@echo $(TAG)Uninstalling httpie$(END)
- pip uninstall --yes httpie &2>/dev/null
@echo "Verifying…"
cd .. && ! python -m httpie --version &2>/dev/null
@echo "Done"
@echo
uninstall-all: uninstall-httpie
@echo $(TAG)Uninstalling httpie requirements$(END)
- pip uninstall --yes pygments requests
@echo $(TAG)Uninstalling development requirements$(END)
- pip uninstall --yes -r $(REQUIREMENTS)

View File

@ -2,9 +2,9 @@
HTTPie: a CLI, cURL-like tool for humans HTTPie: a CLI, cURL-like tool for humans
**************************************** ****************************************
HTTPie (pronounced *aych-tee-tee-pie*) is a **command line HTTP client**. Its HTTPie (pronounced *aitch-tee-tee-pie*) is a **command line HTTP client**.
goal is to make CLI interaction with web services as **human-friendly** as Its goal is to make CLI interaction with web services as **human-friendly**
possible. It provides a simple ``http`` command that allows for sending as possible. It provides a simple ``http`` command that allows for sending
arbitrary HTTP requests using a simple and natural syntax, and displays arbitrary HTTP requests using a simple and natural syntax, and displays
colorized output. HTTPie can be used for **testing, debugging**, and colorized output. HTTPie can be used for **testing, debugging**, and
generally **interacting** with HTTP servers. generally **interacting** with HTTP servers.
@ -588,6 +588,7 @@ Auth Plugins
------------ ------------
* `httpie-oauth <https://github.com/jkbrzt/httpie-oauth>`_: OAuth * `httpie-oauth <https://github.com/jkbrzt/httpie-oauth>`_: OAuth
* `httpie-hmac-auth <https://github.com/guardian/httpie-hmac-auth>`_: HMAC
* `httpie-ntlm <https://github.com/jkbrzt/httpie-ntlm>`_: NTLM (NT LAN Manager) * `httpie-ntlm <https://github.com/jkbrzt/httpie-ntlm>`_: NTLM (NT LAN Manager)
* `httpie-negotiate <https://github.com/ndzou/httpie-negotiate>`_: SPNEGO (GSS Negotiate) * `httpie-negotiate <https://github.com/ndzou/httpie-negotiate>`_: SPNEGO (GSS Negotiate)
* `requests-hawk <https://github.com/mozilla-services/requests-hawk>`_: Hawk * `requests-hawk <https://github.com/mozilla-services/requests-hawk>`_: Hawk

View File

@ -8,8 +8,9 @@ init:
- "ECHO %PYTHON%" - "ECHO %PYTHON%"
- ps: "ls C:/Python*" - ps: "ls C:/Python*"
install: install:
- ps: (new-object net.webclient).DownloadFile('https://raw.github.com/pypa/pip/master/contrib/get-pip.py', 'C:/get-pip.py') # - ps: (new-object net.webclient).DownloadFile('https://bootstrap.pypa.io/get-pip.py', 'C:/get-pip.py')
- "%PYTHON%/python.exe C:/get-pip.py" # - "%PYTHON%/python.exe C:/get-pip.py"
- "%PYTHON%/Scripts/pip.exe install -U pip setuptools"
- "%PYTHON%/Scripts/pip.exe install -e ." - "%PYTHON%/Scripts/pip.exe install -e ."
test_script: test_script:
- "%PYTHON%/Scripts/pip.exe --version" - "%PYTHON%/Scripts/pip.exe --version"

View File

@ -4,7 +4,7 @@ NOTE: the CLI interface may change before reaching v1.0.
""" """
from textwrap import dedent, wrap from textwrap import dedent, wrap
#noinspection PyCompatibility # noinspection PyCompatibility
from argparse import (RawDescriptionHelpFormatter, FileType, from argparse import (RawDescriptionHelpFormatter, FileType,
OPTIONAL, ZERO_OR_MORE, SUPPRESS) OPTIONAL, ZERO_OR_MORE, SUPPRESS)

View File

@ -15,9 +15,10 @@ try:
# https://urllib3.readthedocs.org/en/latest/security.html # https://urllib3.readthedocs.org/en/latest/security.html
urllib3.disable_warnings() urllib3.disable_warnings()
except AttributeError: except AttributeError:
# In some rare cases, the user may have an old version of the requests or urllib3, # In some rare cases, the user may have an old version of the requests
# and there is no method called "disable_warnings." In these cases, we don't need to call # or urllib3, and there is no method called "disable_warnings." In these
# the method. They may get some noisy output but execution shouldn't die. Move on # cases, we don't need to call the method.
# They may get some noisy output but execution shouldn't die. Move on.
pass pass

View File

@ -6,11 +6,11 @@ from httpie import __version__
from httpie.compat import is_windows from httpie.compat import is_windows
DEFAULT_CONFIG_DIR = os.environ.get( DEFAULT_CONFIG_DIR = str(os.environ.get(
'HTTPIE_CONFIG_DIR', 'HTTPIE_CONFIG_DIR',
os.path.expanduser('~/.httpie') if not is_windows else os.path.expanduser('~/.httpie') if not is_windows else
os.path.expandvars(r'%APPDATA%\\httpie') os.path.expandvars(r'%APPDATA%\\httpie')
) ))
class BaseConfigDict(dict): class BaseConfigDict(dict):

View File

@ -378,7 +378,8 @@ class Parser(ArgumentParser):
if self.args.prettify == PRETTY_STDOUT_TTY_ONLY: if self.args.prettify == PRETTY_STDOUT_TTY_ONLY:
self.args.prettify = PRETTY_MAP[ self.args.prettify = PRETTY_MAP[
'all' if self.env.stdout_isatty else 'none'] 'all' if self.env.stdout_isatty else 'none']
elif self.args.prettify and self.env.is_windows: elif (self.args.prettify and self.env.is_windows and
self.args.output_file):
self.error('Only terminal output can be colorized on Windows.') self.error('Only terminal output can be colorized on Windows.')
else: else:
# noinspection PyTypeChecker # noinspection PyTypeChecker
@ -521,7 +522,7 @@ class AuthCredentials(KeyValue):
def _getpass(self, prompt): def _getpass(self, prompt):
# To allow mocking. # To allow mocking.
return getpass.getpass(prompt) return getpass.getpass(str(prompt))
def has_password(self): def has_password(self):
return self.value is not None return self.value is not None
@ -572,7 +573,7 @@ class RequestItemsDict(OrderedDict):
else: else:
super(RequestItemsDict, self).__init__(*args, **kwargs) super(RequestItemsDict, self).__init__(*args, **kwargs)
#noinspection PyMethodOverriding # noinspection PyMethodOverriding
def __setitem__(self, key, value): def __setitem__(self, key, value):
""" If `key` is assigned more than once, `self[key]` holds a """ If `key` is assigned more than once, `self[key]` holds a
`list` of all the values. `list` of all the values.

View File

@ -54,4 +54,3 @@ def humanize_bytes(n, precision=2):
# noinspection PyUnboundLocalVariable # noinspection PyUnboundLocalVariable
return '%.*f %s' % (precision, n / factor, suffix) return '%.*f %s' % (precision, n / factor, suffix)

3
pytest.ini Normal file
View File

@ -0,0 +1,3 @@
[pytest]
addopts = --tb=native
norecursedirs = tests/fixtures

View File

@ -14,7 +14,6 @@ class PyTest(TestCommand):
# and runs the tests with no fancy stuff like parallel execution. # and runs the tests with no fancy stuff like parallel execution.
def finalize_options(self): def finalize_options(self):
TestCommand.finalize_options(self) TestCommand.finalize_options(self)
self.test_suite = True
self.test_args = [ self.test_args = [
'--doctest-modules', '--verbose', '--doctest-modules', '--verbose',
'./httpie', './tests' './httpie', './tests'
@ -40,12 +39,12 @@ install_requires = [
'Pygments>=1.5' 'Pygments>=1.5'
] ]
### Conditional dependencies: # Conditional dependencies:
# sdist # sdist
if not 'bdist_wheel' in sys.argv: if 'bdist_wheel' not in sys.argv:
try: try:
#noinspection PyUnresolvedReferences # noinspection PyUnresolvedReferences
import argparse import argparse
except ImportError: except ImportError:
install_requires.append('argparse>=1.2.1') install_requires.append('argparse>=1.2.1')

View File

@ -38,4 +38,3 @@ with open(BIN_FILE_PATH, 'rb') as f:
BIN_FILE_CONTENT = f.read() BIN_FILE_CONTENT = f.read()
UNICODE = FILE_CONTENT UNICODE = FILE_CONTENT

View File

@ -60,4 +60,3 @@ class TestAuth:
assert args.auth assert args.auth
assert args.auth.key == 'username' assert args.auth.key == 'username'
assert args.auth.value == '' assert args.auth.value == ''

View File

@ -9,7 +9,7 @@ from utils import TESTS_ROOT
def has_docutils(): def has_docutils():
try: try:
#noinspection PyUnresolvedReferences # noinspection PyUnresolvedReferences
import docutils import docutils
return True return True
except ImportError: except ImportError:

View File

@ -64,17 +64,20 @@ class TestSessionFlow(SessionTestBase):
def test_session_update(self, httpbin): def test_session_update(self, httpbin):
self.start_session(httpbin) self.start_session(httpbin)
# Get a response to a request from the original session. # Get a response to a request from the original session.
r2 = http('--session=test', 'GET', httpbin.url + '/get', env=self.env()) r2 = http('--session=test', 'GET', httpbin.url + '/get',
env=self.env())
assert HTTP_OK in r2 assert HTTP_OK in r2
# Make a request modifying the session data. # Make a request modifying the session data.
r3 = http('--follow', '--session=test', '--auth=username:password2', r3 = http('--follow', '--session=test', '--auth=username:password2',
'GET', httpbin.url + '/cookies/set?hello=world2', 'Hello:World2', 'GET', httpbin.url + '/cookies/set?hello=world2',
'Hello:World2',
env=self.env()) env=self.env())
assert HTTP_OK in r3 assert HTTP_OK in r3
# Get a response to a request from the updated session. # Get a response to a request from the updated session.
r4 = http('--session=test', 'GET', httpbin.url + '/get', env=self.env()) r4 = http('--session=test', 'GET', httpbin.url + '/get',
env=self.env())
assert HTTP_OK in r4 assert HTTP_OK in r4
assert r4.json['headers']['Hello'] == 'World2' assert r4.json['headers']['Hello'] == 'World2'
assert r4.json['headers']['Cookie'] == 'hello=world2' assert r4.json['headers']['Cookie'] == 'hello=world2'
@ -84,7 +87,8 @@ class TestSessionFlow(SessionTestBase):
def test_session_read_only(self, httpbin): def test_session_read_only(self, httpbin):
self.start_session(httpbin) self.start_session(httpbin)
# Get a response from the original session. # Get a response from the original session.
r2 = http('--session=test', 'GET', httpbin.url + '/get', env=self.env()) r2 = http('--session=test', 'GET', httpbin.url + '/get',
env=self.env())
assert HTTP_OK in r2 assert HTTP_OK in r2
# Make a request modifying the session data but # Make a request modifying the session data but
@ -96,7 +100,8 @@ class TestSessionFlow(SessionTestBase):
assert HTTP_OK in r3 assert HTTP_OK in r3
# Get a response from the updated session. # Get a response from the updated session.
r4 = http('--session=test', 'GET', httpbin.url + '/get', env=self.env()) r4 = http('--session=test', 'GET', httpbin.url + '/get',
env=self.env())
assert HTTP_OK in r4 assert HTTP_OK in r4
# Origin can differ on Travis. # Origin can differ on Travis.
@ -117,8 +122,8 @@ class TestSession(SessionTestBase):
'If-Unmodified-Since: Sat, 29 Oct 1994 19:43:31 GMT', 'If-Unmodified-Since: Sat, 29 Oct 1994 19:43:31 GMT',
env=self.env()) env=self.env())
assert HTTP_OK in r1 assert HTTP_OK in r1
r2 = http('--session=test', 'GET', httpbin.url + '/get',
r2 = http('--session=test', 'GET', httpbin.url + '/get', env=self.env()) env=self.env())
assert HTTP_OK in r2 assert HTTP_OK in r2
assert no_content_type(r2.json['headers']) assert no_content_type(r2.json['headers'])
assert 'If-Unmodified-Since' not in r2.json['headers'] assert 'If-Unmodified-Since' not in r2.json['headers']

View File

@ -53,7 +53,7 @@ class TestEnvironment(Environment):
stdout_isatty = True stdout_isatty = True
is_windows = False is_windows = False
_shutil = shutil # needed by __del__ (would get gc'd) _shutil_rmtree = shutil.rmtree # needed by __del__ (would get gc'd)
def __init__(self, **kwargs): def __init__(self, **kwargs):
@ -72,7 +72,7 @@ class TestEnvironment(Environment):
def __del__(self): def __del__(self):
if self.delete_config_dir: if self.delete_config_dir:
self._shutil.rmtree(self.config_dir) self._shutil_rmtree(self.config_dir)
def http(*args, **kwargs): def http(*args, **kwargs):

19
tox.ini
View File

@ -1,21 +1,22 @@
# Tox (http://tox.testrun.org/) is a tool for running tests # Tox (http://tox.testrun.org/) is a tool for running tests
# in multiple virtualenvs. # in multiple virtualenvs. See ./CONTRIBUTING.rst
# Run:
# $ pip install -r requirements-dev.txt
# $ tox
[tox] [tox]
envlist = py26, py27, py34, pypy envlist = py26, py27, py35, pypy
[testenv] [testenv]
deps = deps =
mock
pytest pytest
pytest-httpbin>=0.0.6 pytest-httpbin>=0.0.6
commands =
py.test --verbose --doctest-modules --basetemp={envtmpdir} {posargs:./tests ./httpie}
[pytest] commands =
addopts = --tb=native # NOTE: the order of the directories in posargs seems to matter.
# When changed, then many ImportMismatchError exceptions occurrs.
py.test \
--verbose \
--doctest-modules \
{posargs:./httpie ./tests}