This project will require the following steps:
print
and sys.stdout.write
with a higher level abstraction that will be able to either route to
the built in print function or to rich.on
or off
(this
will default to off
)conda info
commandThis refactor will give us a good opportunity to make sure that we are adhering to the principles laid out in this section:
links:
Spinner
and ProgressBar
We also need to think about building an interface for plugins to use and endorse this as the official way to print things for users.
code:
This should be a pretty easy part of the code to refactor. Plus, it will allow us to use rich's nice exception printing:
conda.cli.install
This is a view of everything that needs to be refactored that uses the
conda.cli.install
module. This include commands like, install
, create
and update
.
Components:
These commands should be considerably easier to refactor as there are not nearly
as many moving parts as install
, create
and update
.
Components used:
Just uses print statements
Just uses print statements. It should YAML which could be nicely formatted with rich
Uses print statements
Uses print statements
Components:
Listed as "experimental". Will not refactor.
There's a couple print statements being used in this command (see conda.cli.main_remove
)
Components:
This prints the output from the command that it runs. Does not need refactor
Customized print functions which is a lot like conda info
Components:
Updating this component to support either tqdm or rich will require a refactor of the ProgressBar
class.
I think the best way to do that will be to make a ProgressBarBase class which will serve as an abstract
base class. The ReporterHandler will then have a method for retrieving this class based on the settings
in context.reporters
.
ReporterHandler
The first abstract base class that will be written will be the ReporterHandler
The abstract methods will correspond to display components in conda itself. As of now, they are the following:
detail_view
(currently used in conda info
)string_view
progress_bar
(used in install
, update
, create
)spinner
(used in several commands)package_list
from abc import ABC, abstractmethod
class ReporterHandlerBase(ABC):
"""
Abstract Base Class for all backend/reporters.
"""
@abstractmethod
def detail_view():
"""method to display data in a tabular format"""
...
@abstractmethod
def string_view():
"""method to display data like a regular string, e.g. `conda info --base` simply prints the root prefix string."""
...
class JsonHandler(ReporterHandlerBase):
"""class handling json output"""
def detail_view():
...
def string_view():
...
class StdHandler(ReporterHandlerBase):
"""class handling std output"""
def detail_view():
...
def string_view():
...
OutputHandler
The second abstract base class will concern itself with output. We want to do this because we would eventually like to have a plugin hook that will allow developers add arbitrary output handlers.
This class will be named OutputHandler
and will contain the following abstract methods:
render
from abc import ABC, abstractmethod
import sys
class OutputHandlerBase(ABC):
"""
Base class for all output handlers
"""
@property
@abstractmethod
def name(self) -> str:
"""
Name of the output handler; this will be how this output handler is referenced
in configuration files
"""
...
@abstractmethod
def render(self, output: str) -> None:
"""Render implementation goes here"""
...
class StdoutHandler(OutputHandlerBase):
"""
Handles writing output strings to stdout
"""
@property
def name(self) -> str:
return "stdout"
def render(self, output: str) -> None:
sys.stdout.write(output)
reporters = (
{"backend": "stdout", "output": "stdout"},
{"backend": "json", "output": "file"},
)