httpie-cli/httpie/context.py

129 lines
3.9 KiB
Python
Raw Normal View History

import sys
2020-06-26 18:28:03 +02:00
import os
from pathlib import Path
2020-04-13 17:37:27 +02:00
from typing import IO, Optional
2019-08-29 13:39:42 +02:00
2016-09-06 12:07:52 +02:00
try:
import curses
except ImportError:
curses = None # Compiled w/o curses
from .compat import is_windows
from .config import DEFAULT_CONFIG_DIR, Config, ConfigFileError
from .utils import repr_dict
2016-03-04 18:42:13 +01:00
class Environment:
"""
Information about the execution context
(standard streams, config directory, etc).
By default, it represents the actual environment.
All of the attributes can be overwritten though, which
is used by the test suite to simulate various scenarios.
"""
is_windows: bool = is_windows
config_dir: Path = DEFAULT_CONFIG_DIR
2019-08-29 13:39:42 +02:00
stdin: Optional[IO] = sys.stdin # `None` when closed fd (#791)
stdin_isatty: bool = stdin.isatty() if stdin else False
stdin_encoding: str = None
stdout: IO = sys.stdout
stdout_isatty: bool = stdout.isatty()
stdout_encoding: str = None
stderr: IO = sys.stderr
stderr_isatty: bool = stderr.isatty()
colors = 256
program_name: str = 'http'
if not is_windows:
2016-09-06 12:07:52 +02:00
if curses:
2015-01-23 22:19:02 +01:00
try:
2016-09-06 12:07:52 +02:00
curses.setupterm()
2015-01-23 22:19:02 +01:00
colors = curses.tigetnum('colors')
2016-09-06 12:07:52 +02:00
except curses.error:
pass
else:
# noinspection PyUnresolvedReferences
2020-06-27 18:08:44 +02:00
import colorama.initialise
stdout = colorama.initialise.wrap_stream(
stdout, convert=None, strip=None,
autoreset=True, wrap=True
)
stderr = colorama.initialise.wrap_stream(
stderr, convert=None, strip=None,
autoreset=True, wrap=True
)
del colorama
2020-08-15 15:25:05 +02:00
def __init__(self, devnull=None, **kwargs):
"""
Use keyword arguments to overwrite
any of the class attributes for this instance.
"""
assert all(hasattr(type(self), attr) for attr in kwargs.keys())
self.__dict__.update(**kwargs)
2020-08-15 15:25:05 +02:00
# The original STDERR unaffected by --quieting.
self._orig_stderr = self.stderr
self._devnull = devnull
# Keyword arguments > stream.encoding > default utf8
if self.stdin and self.stdin_encoding is None:
self.stdin_encoding = getattr(
self.stdin, 'encoding', None) or 'utf8'
if self.stdout_encoding is None:
actual_stdout = self.stdout
if is_windows:
# noinspection PyUnresolvedReferences
from colorama import AnsiToWin32
if isinstance(self.stdout, AnsiToWin32):
# noinspection PyUnresolvedReferences
actual_stdout = self.stdout.wrapped
self.stdout_encoding = getattr(
actual_stdout, 'encoding', None) or 'utf8'
2016-03-04 18:42:13 +01:00
def __str__(self):
defaults = dict(type(self).__dict__)
actual = dict(defaults)
actual.update(self.__dict__)
actual['config'] = self.config
2019-08-31 18:00:03 +02:00
return repr_dict({
key: value
2016-03-04 18:42:13 +01:00
for key, value in actual.items()
2019-08-31 18:00:03 +02:00
if not key.startswith('_')
})
2016-03-04 18:42:13 +01:00
def __repr__(self):
2019-08-31 15:17:10 +02:00
return f'<{type(self).__name__} {self}>'
_config: Config = None
@property
def config(self) -> Config:
config = self._config
if not config:
self._config = config = Config(directory=self.config_dir)
if not config.is_new():
try:
config.load()
except ConfigFileError as e:
self.log_error(e, level='warning')
return config
2020-07-15 00:21:57 +02:00
@property
def devnull(self) -> IO:
if self._devnull is None:
self._devnull = open(os.devnull, 'w+')
return self._devnull
@devnull.setter
def devnull(self, value):
self._devnull = value
2020-07-15 00:21:57 +02:00
def log_error(self, msg, level='error'):
assert level in ['error', 'warning']
2020-08-15 15:25:05 +02:00
self._orig_stderr.write(f'\n{self.program_name}: {level}: {msg}\n\n')