2012-07-26 07:37:03 +03:00
|
|
|
"""This module provides the main functionality of HTTPie.
|
|
|
|
|
|
|
|
Invocation flow:
|
|
|
|
|
|
|
|
1. Read, validate and process the input (args, `stdin`).
|
2012-08-03 02:01:15 +03:00
|
|
|
2. Create and send a request.
|
|
|
|
3. Stream, and possibly process and format, the requested parts
|
|
|
|
of the request-response exchange.
|
|
|
|
4. Simultaneously write to `stdout`
|
|
|
|
5. Exit.
|
2012-07-26 07:37:03 +03:00
|
|
|
|
|
|
|
"""
|
2012-07-21 03:59:43 +03:00
|
|
|
import sys
|
2012-08-03 02:01:15 +03:00
|
|
|
import errno
|
2012-07-26 01:26:23 +03:00
|
|
|
|
2012-07-21 03:59:43 +03:00
|
|
|
import requests
|
2012-08-07 15:50:51 +03:00
|
|
|
from httpie import __version__ as httpie_version
|
|
|
|
from requests import __version__ as requests_version
|
|
|
|
from pygments import __version__ as pygments_version
|
2012-07-26 01:26:23 +03:00
|
|
|
|
2012-07-26 07:37:03 +03:00
|
|
|
from .cli import parser
|
2013-01-03 16:12:27 +03:00
|
|
|
from .compat import str, is_py3
|
2012-08-18 00:23:02 +03:00
|
|
|
from .client import get_response
|
2012-08-04 20:12:51 +03:00
|
|
|
from .models import Environment
|
2012-12-05 07:03:18 +03:00
|
|
|
from .output import build_output_stream, write, write_with_colors_win_p3k
|
|
|
|
from . import ExitStatus
|
2012-07-21 03:59:43 +03:00
|
|
|
|
|
|
|
|
2012-12-11 14:54:34 +03:00
|
|
|
def get_exit_status(http_status, follow=False):
|
2012-12-05 07:03:18 +03:00
|
|
|
"""Translate HTTP status code to exit status code."""
|
2012-12-11 14:54:34 +03:00
|
|
|
if 300 <= http_status <= 399 and not follow:
|
2012-07-23 20:35:44 +03:00
|
|
|
# Redirect
|
2012-12-05 07:03:18 +03:00
|
|
|
return ExitStatus.ERROR_HTTP_3XX
|
2012-12-11 14:54:34 +03:00
|
|
|
elif 400 <= http_status <= 499:
|
2012-07-23 20:35:44 +03:00
|
|
|
# Client Error
|
2012-12-05 07:03:18 +03:00
|
|
|
return ExitStatus.ERROR_HTTP_4XX
|
2012-12-11 14:54:34 +03:00
|
|
|
elif 500 <= http_status <= 599:
|
2012-07-23 20:35:44 +03:00
|
|
|
# Server Error
|
2012-12-05 07:03:18 +03:00
|
|
|
return ExitStatus.ERROR_HTTP_5XX
|
2012-07-23 20:35:44 +03:00
|
|
|
else:
|
2012-12-05 07:03:18 +03:00
|
|
|
return ExitStatus.OK
|
2012-07-23 20:35:44 +03:00
|
|
|
|
|
|
|
|
2012-09-17 01:37:36 +03:00
|
|
|
def print_debug_info(env):
|
2012-08-18 05:37:22 +03:00
|
|
|
sys.stderr.writelines([
|
|
|
|
'HTTPie %s\n' % httpie_version,
|
2012-09-17 01:37:36 +03:00
|
|
|
'HTTPie data: %s\n' % env.config.directory,
|
2012-08-18 05:37:22 +03:00
|
|
|
'Requests %s\n' % requests_version,
|
|
|
|
'Pygments %s\n' % pygments_version,
|
|
|
|
'Python %s %s\n' % (sys.version, sys.platform)
|
|
|
|
])
|
|
|
|
|
|
|
|
|
2012-07-21 03:59:43 +03:00
|
|
|
def main(args=sys.argv[1:], env=Environment()):
|
2012-07-26 07:37:03 +03:00
|
|
|
"""Run the main program and write the output to ``env.stdout``.
|
2012-07-23 20:35:44 +03:00
|
|
|
|
2012-12-05 07:03:18 +03:00
|
|
|
Return exit status code.
|
2012-07-23 20:35:44 +03:00
|
|
|
|
|
|
|
"""
|
2012-09-17 03:15:00 +03:00
|
|
|
if env.config.default_options:
|
|
|
|
args = env.config.default_options + args
|
2012-07-30 11:58:16 +03:00
|
|
|
|
2012-12-05 07:27:11 +03:00
|
|
|
def error(msg, *args, **kwargs):
|
2012-08-04 20:12:51 +03:00
|
|
|
msg = msg % args
|
2012-12-05 07:27:11 +03:00
|
|
|
level = kwargs.get('level', 'error')
|
|
|
|
env.stderr.write('http: %s: %s\n' % (level, msg))
|
2012-08-04 20:12:51 +03:00
|
|
|
|
|
|
|
debug = '--debug' in args
|
2012-08-07 15:50:51 +03:00
|
|
|
traceback = debug or '--traceback' in args
|
2012-12-11 14:54:34 +03:00
|
|
|
exit_status = ExitStatus.OK
|
2012-07-30 11:58:16 +03:00
|
|
|
|
2012-08-07 15:50:51 +03:00
|
|
|
if debug:
|
2012-09-17 01:37:36 +03:00
|
|
|
print_debug_info(env)
|
2012-08-18 05:37:22 +03:00
|
|
|
if args == ['--debug']:
|
2012-12-11 14:54:34 +03:00
|
|
|
return exit_status
|
2012-08-07 15:50:51 +03:00
|
|
|
|
2012-07-30 11:58:16 +03:00
|
|
|
try:
|
2012-08-02 00:21:52 +03:00
|
|
|
args = parser.parse_args(args=args, env=env)
|
2012-09-17 03:15:00 +03:00
|
|
|
|
|
|
|
response = get_response(args, config_dir=env.config.directory)
|
2012-08-02 00:21:52 +03:00
|
|
|
|
|
|
|
if args.check_status:
|
2012-12-11 14:54:34 +03:00
|
|
|
exit_status = get_exit_status(response.status_code, args.follow)
|
2012-12-05 07:27:11 +03:00
|
|
|
|
2012-12-11 14:54:34 +03:00
|
|
|
if not env.stdout_isatty and exit_status != ExitStatus.OK:
|
2012-12-05 07:27:11 +03:00
|
|
|
error('HTTP %s %s',
|
|
|
|
response.raw.status,
|
|
|
|
response.raw.reason,
|
|
|
|
level='warning')
|
2012-08-02 00:21:52 +03:00
|
|
|
|
2012-08-17 07:35:18 +03:00
|
|
|
write_kwargs = {
|
2012-12-11 14:54:34 +03:00
|
|
|
'stream': build_output_stream(args, env,
|
|
|
|
response.request,
|
|
|
|
response),
|
2012-08-17 07:35:18 +03:00
|
|
|
'outfile': env.stdout,
|
|
|
|
'flush': env.stdout_isatty or args.stream
|
|
|
|
}
|
2012-08-03 02:01:15 +03:00
|
|
|
try:
|
2012-08-17 07:35:18 +03:00
|
|
|
if env.is_windows and is_py3 and 'colors' in args.prettify:
|
|
|
|
write_with_colors_win_p3k(**write_kwargs)
|
|
|
|
else:
|
|
|
|
write(**write_kwargs)
|
2012-08-03 02:01:15 +03:00
|
|
|
|
|
|
|
except IOError as e:
|
2012-08-07 15:50:51 +03:00
|
|
|
if not traceback and e.errno == errno.EPIPE:
|
2012-08-07 19:22:47 +03:00
|
|
|
# Ignore broken pipes unless --traceback.
|
2012-08-03 02:01:15 +03:00
|
|
|
env.stderr.write('\n')
|
|
|
|
else:
|
2012-08-04 20:12:51 +03:00
|
|
|
raise
|
2012-08-02 00:21:52 +03:00
|
|
|
|
|
|
|
except (KeyboardInterrupt, SystemExit):
|
2012-08-07 15:50:51 +03:00
|
|
|
if traceback:
|
2012-08-03 02:01:15 +03:00
|
|
|
raise
|
2012-08-02 00:21:52 +03:00
|
|
|
env.stderr.write('\n')
|
2012-12-11 14:54:34 +03:00
|
|
|
exit_status = ExitStatus.ERROR
|
2012-12-05 07:03:18 +03:00
|
|
|
|
2012-08-07 19:22:47 +03:00
|
|
|
except requests.Timeout:
|
2012-12-11 14:54:34 +03:00
|
|
|
exit_status = ExitStatus.ERROR_TIMEOUT
|
2012-08-07 19:22:47 +03:00
|
|
|
error('Request timed out (%ss).', args.timeout)
|
2012-12-05 07:03:18 +03:00
|
|
|
|
2012-08-02 00:21:52 +03:00
|
|
|
except Exception as e:
|
2012-12-05 07:03:18 +03:00
|
|
|
# TODO: Better distinction between expected and unexpected errors.
|
|
|
|
# Network errors vs. bugs, etc.
|
2012-08-07 15:50:51 +03:00
|
|
|
if traceback:
|
2012-08-02 00:21:52 +03:00
|
|
|
raise
|
2012-08-04 20:12:51 +03:00
|
|
|
error('%s: %s', type(e).__name__, str(e))
|
2012-12-11 14:54:34 +03:00
|
|
|
exit_status = ExitStatus.ERROR
|
2012-07-24 02:09:14 +03:00
|
|
|
|
2012-12-11 14:54:34 +03:00
|
|
|
return exit_status
|