Skip to content

Commit

Permalink
fix: fail when no style is explicitly configured (don't use the defau…
Browse files Browse the repository at this point in the history
…lt style) (#606)
  • Loading branch information
andreoliwa authored Aug 19, 2023
1 parent db69164 commit f309354
Show file tree
Hide file tree
Showing 9 changed files with 37 additions and 27 deletions.
14 changes: 8 additions & 6 deletions README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -350,6 +350,13 @@ Or install it with pip::
Run
~~~

Nitpick_ will fail if no style is explicitly configured.
Run this command to download and use the opinionated :gitref:`default style file <nitpick-style.toml>`:

nitpick init

You can use it as a template to :ref:`configure-your-own-style`.

To fix and modify your files directly::

nitpick fix
Expand All @@ -358,15 +365,10 @@ To check for errors only::

nitpick check

Nitpick is also a ``flake8`` plugin, so you can run this on a project
with at least one Python (``.py``) file::
Nitpick is also a flake8_ plugin, so you can run this on a project with at least one Python (``.py``) file::

flake8 .

Nitpick will download and use the opinionated `default style file <nitpick-style.toml>`_.

You can use it as a template to configure your own style.

Run as a pre-commit hook
~~~~~~~~~~~~~~~~~~~~~~~~

Expand Down
6 changes: 4 additions & 2 deletions docs/configuration.rst
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,9 @@ Possible configuration files (in order of precedence):
The first file found will be used; the other files will be ignored.

Run the ``nipick init`` CLI command to create a config file (:ref:`cli_cmd_init`).
If no style is configured, Nitpick will fail with an error message.

Run ``nipick init`` to create a config file (:ref:`cli_cmd_init`).

To configure your own style, you can either use ``nitpick init``:

Expand Down Expand Up @@ -217,7 +219,7 @@ Multiple styles

You can also use multiple styles and mix local files and URLs.

Example of usage: the ``[tool.nitpick]`` table on :gitref:`Nitpick's own pyproject.toml <pyproject.toml#L1-L7>`.
Example of usage: the ``[tool.nitpick]`` table on :gitref:`Nitpick's own pyproject.toml <pyproject.toml>`.

.. code-block:: toml
Expand Down
11 changes: 7 additions & 4 deletions docs/quickstart.rst
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,13 @@ Or install it with pip::
Run
---

Nitpick_ will fail if no style is explicitly configured.
Run this command to download and use the opinionated :gitref:`default style file <nitpick-style.toml>`:

nitpick init

You can use it as a template to :ref:`configure-your-own-style`.

To fix and modify your files directly::

nitpick fix
Expand All @@ -48,10 +55,6 @@ Nitpick is also a flake8_ plugin, so you can run this on a project with at least

flake8 .

Nitpick_ will download and use the opinionated :gitref:`default style file <nitpick-style.toml>`.

You can use it as a template to :ref:`configure-your-own-style`.

Run as a pre-commit hook
------------------------

Expand Down
4 changes: 4 additions & 0 deletions src/nitpick/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,10 @@
# Config sections and keys
TOOL_KEY = "tool"
TOOL_NITPICK_KEY = f"{TOOL_KEY}.{PROJECT_NAME}"
RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY = (
f" Run 'nitpick init' or configure a style manually ({', '.join(CONFIG_FILES)})."
f" See {READ_THE_DOCS_URL}configuration.html"
)

# JMESPath expressions
TOOL_NITPICK_JMEX = jmespath.compile(TOOL_NITPICK_KEY)
Expand Down
4 changes: 2 additions & 2 deletions src/nitpick/style/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -115,8 +115,8 @@ def find_initial_styles(self, configured_styles: Sequence[str], base: str | None
chosen_styles = [sorted(paths)[0].expanduser().resolve().as_uri()]
log_message = "Using local style found climbing the directory tree"
else:
chosen_styles = [self.get_default_style_url()]
log_message = "Using default remote Nitpick style"
yield Reporter().make_fuss(StyleViolations.NO_STYLE_CONFIGURED)
return
logger.info(f"{log_message}: {chosen_styles[0]}")

yield from self.include_multiple_styles(
Expand Down
9 changes: 4 additions & 5 deletions src/nitpick/violations.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@

import click

from nitpick.constants import CONFIG_FILES, FLAKE8_PREFIX, READ_THE_DOCS_URL
from nitpick.constants import FLAKE8_PREFIX, RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY

if TYPE_CHECKING:
from nitpick.plugins.info import FileInfo
Expand Down Expand Up @@ -68,16 +68,15 @@ class StyleViolations(ViolationEnum):
INVALID_DATA_TOOL_NITPICK = (1, " has an incorrect style. Invalid data in [{section}]:")
INVALID_TOML = (1, " has an incorrect style. Invalid TOML{exception}")
INVALID_CONFIG = (1, " has an incorrect style. Invalid config:")
NO_STYLE_CONFIGURED = (4, f"No style file configured.{RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY}")


class ProjectViolations(ViolationEnum):
"""Project initialization violations."""

NO_ROOT_DIR = (
101,
"No root directory detected."
f" Create a configuration file ({', '.join(CONFIG_FILES)}) manually, or run 'nitpick init'."
f" See {READ_THE_DOCS_URL}configuration.html",
f"No root directory detected.{RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY}",
)
NO_PYTHON_FILE = (102, "No Python file was found on the root dir and subdir of {root!r}")
MISSING_FILE = (103, " should exist{extra}")
Expand Down Expand Up @@ -141,7 +140,7 @@ def get_counts(cls) -> str:
if cls.fixed:
parts.append(f"✅ {cls.fixed} fixed")
if cls.manual:
parts.append(f"❌ {cls.manual} to change manually")
parts.append(f"❌ {cls.manual} to fix manually")
if not parts:
return "No violations found. ✨ 🍰 ✨"
return f"Violations: {', '.join(parts)}."
2 changes: 1 addition & 1 deletion tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -386,7 +386,7 @@ def cli_run(
return self

if violations:
expected.append(f"Violations: ❌ {violations} to change manually.")
expected.append(f"Violations: ❌ {violations} to fix manually.")
elif expected_str_or_lines:
# If the number of violations was not passed but a list of errors was,
# remove the violation count from the actual results.
Expand Down
10 changes: 5 additions & 5 deletions tests/test_project.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,14 +13,14 @@
PACKAGE_JSON,
PRE_COMMIT_CONFIG_YAML,
PYPROJECT_TOML,
RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY,
SETUP_CFG,
SETUP_PY,
TOX_INI,
)
from nitpick.core import Nitpick
from nitpick.exceptions import QuitComplainingError
from nitpick.project import Configuration, confirm_project_root, find_main_python_file
from nitpick.style import StyleManager
from nitpick.violations import ProjectViolations
from tests.helpers import ProjectMock

Expand Down Expand Up @@ -123,12 +123,12 @@ def test_django_project_structure(tmp_path):


def test_when_no_config_file_the_default_style_is_requested(tmp_path, caplog):
"""There is a root dir (setup.py), but no config file."""
project = ProjectMock(tmp_path, pyproject_toml=False, setup_py=True).api_check(offline=True)
style_url = StyleManager.get_default_style_url()
"""There is a root dir (setup.py), but no style file. The user should explicitly set the style, no default will be used."""
project = ProjectMock(tmp_path, pyproject_toml=False, setup_py=True)
error = f"NIP004 No style file configured.{RUN_NITPICK_INIT_OR_CONFIGURE_STYLE_MANUALLY}"
project.flake8().assert_single_error(error).cli_run(error, exit_code=1)
assert project.nitpick_instance.project.read_configuration() == Configuration(None, [], "")
assert "Config file: none found {}" in caplog.messages
assert f"Using default remote Nitpick style: {style_url} {{}}" in caplog.messages


@pytest.mark.parametrize("config_file", [DOT_NITPICK_TOML, PYPROJECT_TOML])
Expand Down
4 changes: 2 additions & 2 deletions tests/test_violations.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,12 +37,12 @@ def test_reporter():
reporter.increment()
assert reporter.manual == 1
assert reporter.fixed == 0
assert reporter.get_counts() == "Violations: ❌ 1 to change manually."
assert reporter.get_counts() == "Violations: ❌ 1 to fix manually."

reporter.increment(True)
assert reporter.manual == 1
assert reporter.fixed == 1
assert reporter.get_counts() == "Violations: ✅ 1 fixed, ❌ 1 to change manually."
assert reporter.get_counts() == "Violations: ✅ 1 fixed, ❌ 1 to fix manually."

reporter.reset()
assert reporter.manual == 0
Expand Down

0 comments on commit f309354

Please sign in to comment.