mirror of
https://github.com/HorlogeSkynet/archey4
synced 2024-12-28 04:00:13 +01:00
191 lines
7.7 KiB
Python
191 lines
7.7 KiB
Python
"""
|
|
Output class file.
|
|
It supports entries lazy-insertion, logo detection, and final printing.
|
|
"""
|
|
|
|
import os
|
|
import sys
|
|
from shutil import get_terminal_size
|
|
from textwrap import TextWrapper
|
|
from typing import cast
|
|
|
|
from archey.api import API
|
|
from archey.colors import ANSI_ECMA_REGEXP, Colors, Style
|
|
from archey.configuration import Configuration
|
|
from archey.distributions import Distributions
|
|
from archey.entry import Entry
|
|
from archey.exceptions import ArcheyException
|
|
from archey.logos import get_logo_width, lazy_load_logo_module
|
|
|
|
|
|
class Output:
|
|
"""
|
|
This is the object handling output entries populating.
|
|
It also handles the logo choice based on some system detections.
|
|
"""
|
|
|
|
__logo_right_padding = " "
|
|
|
|
def __init__(self, **kwargs):
|
|
configuration = Configuration()
|
|
|
|
# Fetches passed arguments.
|
|
self._format_to_json = kwargs.get("format_to_json")
|
|
preferred_logo_style = (
|
|
kwargs.get("preferred_logo_style") or configuration.get("logo_style") or ""
|
|
).upper()
|
|
|
|
# If logo shouldn't be displayed, don't load any module and reset right padding
|
|
if preferred_logo_style == "NONE":
|
|
self._logo, self._colors = [], []
|
|
self.__logo_right_padding = ""
|
|
else:
|
|
try:
|
|
# If set, force the distribution to `preferred_distribution` argument.
|
|
distribution = Distributions(kwargs.get("preferred_distribution"))
|
|
except ValueError:
|
|
# If not (or unknown), run distribution detection.
|
|
distribution = Distributions.get_local()
|
|
|
|
# Retrieve distribution's logo module before copying and DRY-ing its attributes.
|
|
logo_module = lazy_load_logo_module(distribution.value)
|
|
|
|
# If set and available, fetch an alternative logo style from module.
|
|
if preferred_logo_style and hasattr(logo_module, f"LOGO_{preferred_logo_style}"):
|
|
self._logo = getattr(logo_module, f"LOGO_{preferred_logo_style}").copy()
|
|
self._colors = getattr(logo_module, f"COLORS_{preferred_logo_style}").copy()
|
|
else:
|
|
self._logo, self._colors = logo_module.LOGO.copy(), logo_module.COLORS.copy()
|
|
|
|
# If `os-release`'s `ANSI_COLOR` option is set, honor it.
|
|
ansi_color = Distributions.get_ansi_color()
|
|
if ansi_color and configuration.get("honor_ansi_color"):
|
|
# Replace each Archey integrated colors by `ANSI_COLOR`.
|
|
self._colors = len(self._colors) * [Style.escape_code_from_attrs(ansi_color)]
|
|
|
|
entries_color = configuration.get("entries_color")
|
|
if entries_color:
|
|
self._entries_color = Style.escape_code_from_attrs(entries_color)
|
|
elif self._colors:
|
|
self._entries_color = str(self._colors[0])
|
|
else:
|
|
self._entries_color = ""
|
|
|
|
# Each entry will be added to this list
|
|
self._entries = []
|
|
# Each class output will be added in the list below afterwards
|
|
self._results = []
|
|
|
|
def add_entry(self, module: Entry) -> None:
|
|
"""Append an entry to the list of entries to output"""
|
|
self._entries.append(module)
|
|
|
|
def append(self, key: str, value) -> None:
|
|
"""Append a pre-formatted entry to the final output content"""
|
|
self._results.append(f"{self._entries_color}{key}:{Colors.CLEAR} {value}")
|
|
|
|
def output(self) -> None:
|
|
"""
|
|
Main `Output`'s `output` method.
|
|
First we get entries to add their outputs to the results and then
|
|
calls specific `output` methods based (for instance) on preferred format.
|
|
"""
|
|
if self._format_to_json:
|
|
self._output_json()
|
|
else:
|
|
# Iterate through the entries and run their output method to add their content.
|
|
for entry in self._entries:
|
|
entry.output(self)
|
|
self._output_text()
|
|
|
|
def _output_json(self) -> None:
|
|
"""
|
|
Finally outputs entries data to JSON format.
|
|
See `archey.api` for further documentation.
|
|
"""
|
|
print(API(self._entries).json_serialization(indent=cast(int, self._format_to_json) - 1))
|
|
|
|
def _output_text(self) -> None:
|
|
"""
|
|
Finally render the output entries.
|
|
It handles text centering additionally to value and colors replacing.
|
|
"""
|
|
# Compute the effective logo "width" from the loaded ASCII art.
|
|
logo_width = get_logo_width(self._logo, len(self._colors))
|
|
|
|
# Let's center the entries and the logo (handles odd numbers)
|
|
height_diff = len(self._logo) - len(self._results)
|
|
if height_diff >= 0:
|
|
self._results[0:0] = [""] * (height_diff // 2)
|
|
self._results.extend([""] * (len(self._logo) - len(self._results)))
|
|
else:
|
|
colored_empty_line = [(str(self._colors[0]) if self._colors else "") + " " * logo_width]
|
|
self._logo[0:0] = colored_empty_line * (-height_diff // 2)
|
|
self._logo.extend(colored_empty_line * (len(self._results) - len(self._logo)))
|
|
|
|
# When writing to a pipe (for instance), prevent `TextWrapper` from truncating output.
|
|
if not sys.stdout.isatty():
|
|
text_width = cast(int, float("inf"))
|
|
else:
|
|
text_width = get_terminal_size().columns - logo_width - len(self.__logo_right_padding)
|
|
|
|
text_wrapper = TextWrapper(
|
|
width=text_width,
|
|
expand_tabs=False,
|
|
replace_whitespace=False,
|
|
drop_whitespace=False,
|
|
break_on_hyphens=False,
|
|
max_lines=1,
|
|
placeholder="...",
|
|
)
|
|
placeholder_length = len(text_wrapper.placeholder)
|
|
|
|
# Using `TextWrapper`, shortens each entry to remove any line overlapping
|
|
for i, entry in enumerate(self._results):
|
|
# Shortens the entry according to the terminal width.
|
|
# We have to remove any ANSI color, or the result would be skewed.
|
|
wrapped_entry = text_wrapper.fill(Style.remove_colors(entry))
|
|
placeholder_offset = (
|
|
placeholder_length if wrapped_entry.endswith(text_wrapper.placeholder) else 0
|
|
)
|
|
|
|
# By using previous positions, re-inserts ANSI colors back in the wrapped string.
|
|
for color_match in ANSI_ECMA_REGEXP.finditer(entry):
|
|
match_index = color_match.start()
|
|
if match_index <= len(wrapped_entry) - placeholder_offset:
|
|
wrapped_entry = (
|
|
wrapped_entry[:match_index]
|
|
+ color_match.group()
|
|
+ wrapped_entry[match_index:]
|
|
)
|
|
|
|
# Add a color reset character before the placeholder (if any).
|
|
# Rationale :
|
|
# We cannot set `Colors.CLEAR` in the placeholder as it would skew its internals.
|
|
if placeholder_offset:
|
|
wrapped_entry = (
|
|
wrapped_entry[:-placeholder_length]
|
|
+ str(Colors.CLEAR)
|
|
+ wrapped_entry[-placeholder_length:]
|
|
)
|
|
|
|
self._results[i] = wrapped_entry
|
|
|
|
# Merge entry results to the distribution logo.
|
|
logo_with_entries = os.linesep.join(
|
|
[
|
|
f"{logo_part}{self.__logo_right_padding}{entry_part}"
|
|
for logo_part, entry_part in zip(self._logo, self._results)
|
|
]
|
|
)
|
|
|
|
try:
|
|
print(logo_with_entries.format(c=self._colors) + str(Colors.CLEAR))
|
|
except UnicodeError as unicode_error:
|
|
raise ArcheyException(
|
|
"""\
|
|
Your locale or TTY does not seem to support UTF-8 encoding.
|
|
Please disable Unicode within your configuration file.\
|
|
"""
|
|
) from unicode_error
|