mirror of
https://github.com/httpie/cli.git
synced 2025-01-10 00:28:12 +02:00
269 lines
8.0 KiB
Python
269 lines
8.0 KiB
Python
"""Output processing and formatting.
|
|
|
|
"""
|
|
import re
|
|
import json
|
|
|
|
import pygments
|
|
from pygments import token, lexer
|
|
from pygments.styles import get_style_by_name, STYLE_MAP
|
|
from pygments.lexers import get_lexer_for_mimetype, get_lexer_by_name
|
|
from pygments.formatters.terminal import TerminalFormatter
|
|
from pygments.formatters.terminal256 import Terminal256Formatter
|
|
from pygments.util import ClassNotFound
|
|
from requests.compat import is_windows
|
|
|
|
from .solarized import Solarized256Style
|
|
from .models import Environment
|
|
|
|
|
|
DEFAULT_STYLE = 'solarized'
|
|
AVAILABLE_STYLES = [DEFAULT_STYLE] + list(STYLE_MAP.keys())
|
|
BINARY_SUPPRESSED_NOTICE = (
|
|
'+-----------------------------------------+\n'
|
|
'| NOTE: binary data not shown in terminal |\n'
|
|
'+-----------------------------------------+'
|
|
)
|
|
|
|
|
|
def formatted_stream(msg, prettifier=None, with_headers=True, with_body=True,
|
|
env=Environment()):
|
|
"""Return an iterator yielding `bytes` representing `msg`
|
|
(a `models.HTTPMessage` subclass).
|
|
|
|
The body can be binary so we always yield `bytes`.
|
|
|
|
If `prettifier` is set or the output is a terminal then a binary
|
|
body is not included in the output and is replaced with notice.
|
|
|
|
Generally, when the `stdout` is redirected, the output matches the actual
|
|
message as much as possible (formatting and character encoding-wise).
|
|
When `--pretty` is set (or implied), or when the output is a terminal,
|
|
then we prefer readability over precision.
|
|
|
|
"""
|
|
# Output encoding.
|
|
if env.stdout_isatty:
|
|
# Use encoding suitable for the terminal. Unsupported characters
|
|
# will be replaced in the output.
|
|
errors = 'replace'
|
|
output_encoding = getattr(env.stdout, 'encoding', None)
|
|
else:
|
|
# Preserve the message encoding.
|
|
errors = 'strict'
|
|
output_encoding = msg.encoding
|
|
if not output_encoding:
|
|
# Default to utf8
|
|
output_encoding = 'utf8'
|
|
|
|
if prettifier:
|
|
env.init_colors()
|
|
|
|
if with_headers:
|
|
headers = '\n'.join([msg.line, msg.headers])
|
|
|
|
if prettifier:
|
|
headers = prettifier.process_headers(headers)
|
|
|
|
yield headers.encode(output_encoding, errors).strip()
|
|
|
|
if with_body:
|
|
|
|
prefix = b'\n\n' if with_headers else None
|
|
|
|
if not (env.stdout_isatty or prettifier):
|
|
# Verbatim body even if it's binary.
|
|
for body_chunk in msg:
|
|
if prefix:
|
|
yield prefix
|
|
prefix = None
|
|
yield body_chunk
|
|
elif msg.body:
|
|
try:
|
|
body = msg.body.decode(msg.encoding)
|
|
except UnicodeDecodeError:
|
|
# Suppress binary data.
|
|
body = BINARY_SUPPRESSED_NOTICE.encode(output_encoding)
|
|
if not with_headers:
|
|
yield b'\n'
|
|
else:
|
|
if prettifier and msg.content_type:
|
|
body = prettifier.process_body(
|
|
body, msg.content_type).strip()
|
|
|
|
body = body.encode(output_encoding, errors)
|
|
if prefix:
|
|
yield prefix
|
|
yield body
|
|
|
|
|
|
class HTTPLexer(lexer.RegexLexer):
|
|
"""Simplified HTTP lexer for Pygments.
|
|
|
|
It only operates on headers and provides a stronger contrast between
|
|
their names and values than the original one bundled with Pygments
|
|
(`pygments.lexers.text import HttpLexer`), especially when
|
|
Solarized color scheme is used.
|
|
|
|
"""
|
|
name = 'HTTP'
|
|
aliases = ['http']
|
|
filenames = ['*.http']
|
|
tokens = {
|
|
'root': [
|
|
|
|
# Request-Line
|
|
(r'([A-Z]+)( +)([^ ]+)( +)(HTTP)(/)(\d+\.\d+)',
|
|
lexer.bygroups(
|
|
token.Name.Function,
|
|
token.Text,
|
|
token.Name.Namespace,
|
|
token.Text,
|
|
token.Keyword.Reserved,
|
|
token.Operator,
|
|
token.Number
|
|
)),
|
|
|
|
# Response Status-Line
|
|
(r'(HTTP)(/)(\d+\.\d+)( +)(\d{3})( +)(.+)',
|
|
lexer.bygroups(
|
|
token.Keyword.Reserved, # 'HTTP'
|
|
token.Operator, # '/'
|
|
token.Number, # Version
|
|
token.Text,
|
|
token.Number, # Status code
|
|
token.Text,
|
|
token.Name.Exception, # Reason
|
|
)),
|
|
|
|
# Header
|
|
(r'(.*?)( *)(:)( *)(.+)', lexer.bygroups(
|
|
token.Name.Attribute, # Name
|
|
token.Text,
|
|
token.Operator, # Colon
|
|
token.Text,
|
|
token.String # Value
|
|
))
|
|
]}
|
|
|
|
|
|
class BaseProcessor(object):
|
|
|
|
enabled = True
|
|
|
|
def __init__(self, env, **kwargs):
|
|
self.env = env
|
|
self.kwargs = kwargs
|
|
|
|
def process_headers(self, headers):
|
|
return headers
|
|
|
|
def process_body(self, content, content_type, subtype):
|
|
"""Return processed `content`.
|
|
|
|
:param content: `str`
|
|
:param content_type: full content type, e.g., 'application/atom+xml'
|
|
:param subtype: e.g., 'xml'
|
|
|
|
"""
|
|
return content
|
|
|
|
|
|
class JSONProcessor(BaseProcessor):
|
|
|
|
def process_body(self, content, content_type, subtype):
|
|
if subtype == 'json':
|
|
try:
|
|
# Indent the JSON data, sort keys by name, and
|
|
# avoid unicode escapes to improve readability.
|
|
content = json.dumps(json.loads(content),
|
|
sort_keys=True,
|
|
ensure_ascii=False,
|
|
indent=4)
|
|
except ValueError:
|
|
# Invalid JSON but we don't care.
|
|
pass
|
|
return content
|
|
|
|
|
|
class PygmentsProcessor(BaseProcessor):
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super(PygmentsProcessor, self).__init__(*args, **kwargs)
|
|
|
|
if not self.env.colors:
|
|
self.enabled = False
|
|
return
|
|
|
|
try:
|
|
style = get_style_by_name(
|
|
self.kwargs.get('pygments_style', DEFAULT_STYLE))
|
|
except ClassNotFound:
|
|
style = Solarized256Style
|
|
|
|
if is_windows or self.env.colors == 256:
|
|
fmt_class = Terminal256Formatter
|
|
else:
|
|
fmt_class = TerminalFormatter
|
|
self.formatter = fmt_class(style=style)
|
|
|
|
def process_headers(self, headers):
|
|
return pygments.highlight(
|
|
headers, HTTPLexer(), self.formatter)
|
|
|
|
def process_body(self, content, content_type, subtype):
|
|
try:
|
|
try:
|
|
lexer = get_lexer_for_mimetype(content_type)
|
|
except ClassNotFound:
|
|
lexer = get_lexer_by_name(subtype)
|
|
except ClassNotFound:
|
|
pass
|
|
else:
|
|
content = pygments.highlight(content, lexer, self.formatter)
|
|
return content
|
|
|
|
|
|
class HeadersProcessor(BaseProcessor):
|
|
"""
|
|
Sorts headers by name retaining relative order of multiple headers
|
|
with the same name.
|
|
|
|
"""
|
|
def process_headers(self, headers):
|
|
lines = headers.splitlines()
|
|
headers = sorted(lines[1:], key=lambda h: h.split(':')[0])
|
|
return '\n'.join(lines[:1] + headers)
|
|
|
|
|
|
class OutputProcessor(object):
|
|
|
|
installed_processors = [
|
|
JSONProcessor,
|
|
HeadersProcessor,
|
|
PygmentsProcessor
|
|
]
|
|
|
|
def __init__(self, env, **kwargs):
|
|
processors = [
|
|
cls(env, **kwargs)
|
|
for cls in self.installed_processors
|
|
]
|
|
self.processors = [p for p in processors if p.enabled]
|
|
|
|
def process_headers(self, headers):
|
|
for processor in self.processors:
|
|
headers = processor.process_headers(headers)
|
|
return headers
|
|
|
|
def process_body(self, content, content_type):
|
|
# e.g., 'application/atom+xml'
|
|
content_type = content_type.split(';')[0]
|
|
# e.g., 'xml'
|
|
subtype = content_type.split('/')[-1].split('+')[-1]
|
|
|
|
for processor in self.processors:
|
|
content = processor.process_body(content, content_type, subtype)
|
|
|
|
return content
|