2012-02-25 13:39:38 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import json
|
|
|
|
import argparse
|
|
|
|
from collections import namedtuple
|
|
|
|
import requests
|
|
|
|
from requests.structures import CaseInsensitiveDict
|
2012-02-26 01:37:28 +01:00
|
|
|
from . import pretty
|
2012-02-28 17:14:31 +01:00
|
|
|
from . import __version__ as version
|
|
|
|
from . import __doc__ as doc
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
|
2012-02-28 17:14:31 +01:00
|
|
|
DEFAULT_UA = 'HTTPie/%s' % version
|
2012-02-25 13:39:38 +01:00
|
|
|
SEP_COMMON = ':'
|
|
|
|
SEP_DATA = '='
|
|
|
|
TYPE_FORM = 'application/x-www-form-urlencoded; charset=utf-8'
|
|
|
|
TYPE_JSON = 'application/json; charset=utf-8'
|
2012-03-04 01:36:39 +01:00
|
|
|
PRETTIFY_STDOUT_TTY_ONLY = object()
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
|
|
|
|
KeyValue = namedtuple('KeyValue', ['key', 'value', 'sep'])
|
|
|
|
|
|
|
|
|
|
|
|
class KeyValueType(object):
|
|
|
|
|
|
|
|
def __init__(self, separators):
|
|
|
|
self.separators = separators
|
|
|
|
|
|
|
|
def __call__(self, string):
|
2012-02-28 18:06:21 +01:00
|
|
|
found = dict((string.find(sep), sep)
|
|
|
|
for sep in self.separators
|
|
|
|
if string.find(sep) != -1)
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
if not found:
|
|
|
|
raise argparse.ArgumentTypeError(
|
|
|
|
'"%s" is not a valid value' % string)
|
|
|
|
sep = found[min(found.keys())]
|
|
|
|
key, value = string.split(sep, 1)
|
|
|
|
return KeyValue(key=key, value=value, sep=sep)
|
|
|
|
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(
|
2012-02-28 17:14:31 +01:00
|
|
|
description=doc.strip())
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
|
|
|
|
# Content type.
|
2012-02-26 01:37:28 +01:00
|
|
|
group_type = parser.add_mutually_exclusive_group(required=False)
|
|
|
|
group_type.add_argument('--json', '-j', action='store_true',
|
2012-02-25 13:39:38 +01:00
|
|
|
help='Serialize data items as a JSON object and set'
|
|
|
|
' Content-Type to application/json, if not specified.')
|
2012-02-26 01:37:28 +01:00
|
|
|
group_type.add_argument('--form', '-f', action='store_true',
|
2012-02-25 13:39:38 +01:00
|
|
|
help='Serialize data items as form values and set'
|
|
|
|
' Content-Type to application/x-www-form-urlencoded,'
|
|
|
|
' if not specified.')
|
|
|
|
|
2012-02-26 01:37:28 +01:00
|
|
|
# Output options.
|
2012-02-26 16:13:12 +01:00
|
|
|
parser.add_argument('--traceback', action='store_true', default=False,
|
|
|
|
help='Print a full exception traceback should one'
|
|
|
|
' be raised by `requests`.')
|
2012-03-04 01:36:39 +01:00
|
|
|
group_pretty = parser.add_mutually_exclusive_group(required=False)
|
|
|
|
group_pretty.add_argument('--pretty', '-p', dest='prettify', action='store_true',
|
|
|
|
default=PRETTIFY_STDOUT_TTY_ONLY,
|
|
|
|
help='If stdout is a terminal, '
|
|
|
|
' the response is prettified by default (colorized and'
|
|
|
|
' indented if it is JSON). This flag ensures'
|
|
|
|
' prettifying even when stdout is redirected.')
|
|
|
|
group_pretty.add_argument('--ugly', '-u', help='Do not prettify the response.',
|
|
|
|
dest='prettify', action='store_false')
|
2012-02-26 01:37:28 +01:00
|
|
|
group_only = parser.add_mutually_exclusive_group(required=False)
|
|
|
|
group_only.add_argument('--headers', '-t', dest='print_body',
|
|
|
|
action='store_false', default=True,
|
|
|
|
help='Print only the response headers.')
|
|
|
|
group_only.add_argument('--body', '-b', dest='print_headers',
|
|
|
|
action='store_false', default=True,
|
|
|
|
help='Print only the response body.')
|
2012-03-02 01:40:40 +01:00
|
|
|
parser.add_argument('--style', '-s', dest='style', default='solarized', metavar='STYLE',
|
|
|
|
choices=pretty.AVAILABLE_STYLES,
|
|
|
|
help='Output coloring style, one of %s. Defaults to solarized.'
|
|
|
|
% ', '.join(sorted(pretty.AVAILABLE_STYLES)))
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
# ``requests.request`` keyword arguments.
|
2012-02-27 11:54:41 +01:00
|
|
|
parser.add_argument('--auth', '-a', help='username:password',
|
2012-02-25 13:39:38 +01:00
|
|
|
type=KeyValueType(SEP_COMMON))
|
|
|
|
parser.add_argument('--verify',
|
|
|
|
help='Set to "yes" to check the host\'s SSL certificate.'
|
|
|
|
' You can also pass the path to a CA_BUNDLE'
|
|
|
|
' file for private certs. You can also set '
|
|
|
|
'the REQUESTS_CA_BUNDLE environment variable.')
|
|
|
|
parser.add_argument('--proxy', default=[], action='append',
|
|
|
|
type=KeyValueType(SEP_COMMON),
|
|
|
|
help='String mapping protocol to the URL of the proxy'
|
|
|
|
' (e.g. http:foo.bar:3128).')
|
|
|
|
parser.add_argument('--allow-redirects', default=False, action='store_true',
|
|
|
|
help='Set this flag if full redirects are allowed'
|
|
|
|
' (e.g. re-POST-ing of data at new ``Location``)')
|
|
|
|
parser.add_argument('--file', metavar='PATH', type=argparse.FileType(),
|
|
|
|
default=[], action='append',
|
|
|
|
help='File to multipart upload')
|
|
|
|
parser.add_argument('--timeout', type=float,
|
|
|
|
help='Float describes the timeout of the request'
|
|
|
|
' (Use socket.setdefaulttimeout() as fallback).')
|
|
|
|
|
|
|
|
# Positional arguments.
|
2012-03-04 01:36:39 +01:00
|
|
|
parser.add_argument('method', metavar='METHOD',
|
2012-02-25 13:39:38 +01:00
|
|
|
help='HTTP method to be used for the request'
|
|
|
|
' (GET, POST, PUT, DELETE, PATCH, ...).')
|
|
|
|
parser.add_argument('url', metavar='URL',
|
|
|
|
help='Protocol defaults to http:// if the'
|
|
|
|
' URL does not include it.')
|
2012-03-04 01:36:39 +01:00
|
|
|
parser.add_argument('items', nargs='*',
|
2012-02-25 13:39:38 +01:00
|
|
|
type=KeyValueType([SEP_COMMON, SEP_DATA]),
|
|
|
|
help='HTTP header (key:value) or data field (key=value)')
|
|
|
|
|
|
|
|
|
2012-03-02 01:40:40 +01:00
|
|
|
def main(args=None,
|
|
|
|
stdin=sys.stdin,
|
2012-03-02 02:36:21 +01:00
|
|
|
stdin_isatty=sys.stdin.isatty(),
|
2012-03-02 01:40:40 +01:00
|
|
|
stdout=sys.stdout,
|
|
|
|
stdout_isatty=sys.stdout.isatty()):
|
|
|
|
|
|
|
|
args = parser.parse_args(args if args is not None else sys.argv[1:])
|
2012-03-04 01:36:39 +01:00
|
|
|
do_prettify = (args.prettify is True or
|
|
|
|
(args.prettify == PRETTIFY_STDOUT_TTY_ONLY and stdout_isatty))
|
2012-02-25 13:39:38 +01:00
|
|
|
# Parse request headers and data from the command line.
|
|
|
|
headers = CaseInsensitiveDict()
|
|
|
|
headers['User-Agent'] = DEFAULT_UA
|
|
|
|
data = {}
|
|
|
|
for item in args.items:
|
|
|
|
if item.sep == SEP_COMMON:
|
|
|
|
target = headers
|
|
|
|
else:
|
2012-03-02 01:40:40 +01:00
|
|
|
if not stdin_isatty:
|
2012-02-25 13:39:38 +01:00
|
|
|
parser.error('Request body (stdin) and request '
|
|
|
|
'data (key=value) cannot be mixed.')
|
|
|
|
target = data
|
|
|
|
target[item.key] = item.value
|
|
|
|
|
2012-03-02 01:40:40 +01:00
|
|
|
if not stdin_isatty:
|
|
|
|
data = stdin.read()
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
# JSON/Form content type.
|
|
|
|
if args.json or (not args.form and data):
|
2012-03-02 01:40:40 +01:00
|
|
|
if stdin_isatty:
|
2012-02-25 13:39:38 +01:00
|
|
|
data = json.dumps(data)
|
|
|
|
if 'Content-Type' not in headers and (data or args.json):
|
|
|
|
headers['Content-Type'] = TYPE_JSON
|
|
|
|
elif 'Content-Type' not in headers:
|
|
|
|
headers['Content-Type'] = TYPE_FORM
|
|
|
|
|
|
|
|
# Fire the request.
|
2012-02-26 16:13:12 +01:00
|
|
|
try:
|
|
|
|
response = requests.request(
|
|
|
|
method=args.method.lower(),
|
|
|
|
url=args.url if '://' in args.url else 'http://%s' % args.url,
|
|
|
|
headers=headers,
|
|
|
|
data=data,
|
|
|
|
verify=True if args.verify == 'yes' else args.verify,
|
|
|
|
timeout=args.timeout,
|
|
|
|
auth=(args.auth.key, args.auth.value) if args.auth else None,
|
2012-02-28 14:33:33 +01:00
|
|
|
proxies=dict((p.key, p.value) for p in args.proxy),
|
|
|
|
files=dict((os.path.basename(f.name), f) for f in args.file),
|
2012-03-03 20:54:53 +01:00
|
|
|
allow_redirects=args.allow_redirects,
|
2012-02-26 16:13:12 +01:00
|
|
|
)
|
|
|
|
except (KeyboardInterrupt, SystemExit) as e:
|
|
|
|
sys.stderr.write('\n')
|
|
|
|
sys.exit(1)
|
|
|
|
except Exception as e:
|
|
|
|
if args.traceback:
|
|
|
|
raise
|
|
|
|
sys.stderr.write(str(e.message) + '\n')
|
|
|
|
sys.exit(1)
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
# Display the response.
|
2012-02-28 18:06:21 +01:00
|
|
|
encoding = response.encoding or 'ISO-8859-1'
|
2012-02-25 13:39:38 +01:00
|
|
|
original = response.raw._original_response
|
2012-02-26 01:37:28 +01:00
|
|
|
status_line, headers, body = (
|
2012-03-02 02:36:21 +01:00
|
|
|
'HTTP/{version} {status} {reason}'.format(
|
2012-02-25 13:39:38 +01:00
|
|
|
version='.'.join(str(original.version)),
|
|
|
|
status=original.status, reason=original.reason,
|
|
|
|
),
|
2012-02-28 18:06:21 +01:00
|
|
|
str(original.msg).decode(encoding),
|
|
|
|
response.content.decode(encoding) if response.content else u''
|
2012-02-25 13:39:38 +01:00
|
|
|
)
|
|
|
|
|
2012-03-04 01:36:39 +01:00
|
|
|
if do_prettify:
|
2012-03-02 01:40:40 +01:00
|
|
|
prettify = pretty.PrettyHttp(args.style)
|
2012-02-26 01:37:28 +01:00
|
|
|
if args.print_headers:
|
2012-03-02 01:40:40 +01:00
|
|
|
status_line = prettify.headers(status_line).strip()
|
|
|
|
headers = prettify.headers(headers)
|
|
|
|
if args.print_body and 'content-type' in response.headers:
|
|
|
|
body = prettify.body(body, response.headers['content-type'])
|
2012-02-26 01:37:28 +01:00
|
|
|
|
|
|
|
if args.print_headers:
|
2012-03-02 01:40:40 +01:00
|
|
|
stdout.write(status_line)
|
|
|
|
stdout.write('\n')
|
2012-03-02 02:36:21 +01:00
|
|
|
stdout.write(headers.encode('utf-8'))
|
2012-03-02 09:02:50 +01:00
|
|
|
stdout.write('\n\n')
|
2012-02-26 01:37:28 +01:00
|
|
|
if args.print_body:
|
2012-03-02 02:36:21 +01:00
|
|
|
stdout.write(body.encode('utf-8'))
|
|
|
|
stdout.write('\n')
|
2012-02-25 13:39:38 +01:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|