-
Notifications
You must be signed in to change notification settings - Fork 1.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Ruff incorrectly matches .gitignore rules with sub-directories and has issues with respect-gitignore = false
#13692
Comments
Hi @mihnatenko
Can you explain me why you think that Ruff shouldn't ignore the file? What I understand from the description is that the file gets ignored because you ignored the entire
The That's why the You could consider adding a |
Hello @MichaReiser
Since git doesn't ignore it. You can verify that by mkdir ruff-ignore-subdir-bug && cd ruff-ignore-subdir-bug
git init
echo "backend/log" > .gitignore
mkdir -p backend/log
mkdir -p backend/src/subdir/log
echo "# no docstring, but should be ignored" > backend/log/ignore_me.py
echo "# no docstring, should NOT be ignored" > backend/src/subdir/log/do_not_ignore_me.py
Git does ignore /tmp/ruff-ignore-subdir-bug ❯ git add .gitignore backend/
/tmp/ruff-ignore-subdir-bug ❯ git status
On branch main
No commits yet
Changes to be committed:
(use "git rm --cached <file>..." to unstage)
new file: .gitignore
new file: backend/src/subdir/log/do_not_ignore_me.py And this in my opinion is expected behavior, since if you would need to ignore any arbitrarily nested But ruff ignores both /tmp/ruff-ignore-subdir-bug ❯ ruff check --select=D100 backend
warning: No Python files found under the given path(s)
All checks passed!
/tmp/ruff-ignore-subdir-bug ❯ cd backend
/tmp/ruff-ignore-subdir-bug/backend ❯ ruff check --select=D100
warning: No Python files found under the given path(s)
All checks passed! So does ripgrep. But ripgrep does that in a different manner. It works correctly when executed from root of the repository, but incorrectly when executed from backend: /tmp/ruff-ignore-subdir-bug ❯ rg "be ignored"
backend/src/subdir/log/do_not_ignore_me.py
1:# no docstring, so should NOT be ignored
/tmp/ruff-ignore-subdir-bug ❯ cd backend
/tmp/ruff-ignore-subdir-bug/backend ❯ rg "be ignored"
rg: No files were searched, which means ripgrep probably applied a filter you didn't expect.
Running with --debug will show why files are being skipped. I presume both ruff and ripgrep do that incorrectly, since they use the same |
Thanks for the example. I tried reproducing the problem you're mentioning but running ruff in the directory when enabling the pydocstyle rules does flag the missing docstring in uvx ruff check . -v --select D
[2024-10-14][14:10:15][ruff::resolve][DEBUG] Using configuration file (via parent) at: /home/micha/astral/test/pyproject.toml
warning: `one-blank-line-before-class` (D203) and `no-blank-line-before-class` (D211) are incompatible. Ignoring `one-blank-line-before-class`.
warning: `multi-line-summary-first-line` (D212) and `multi-line-summary-second-line` (D213) are incompatible. Ignoring `multi-line-summary-second-line`.
[2024-10-14][14:10:15][ignore::gitignore][DEBUG] opened gitignore file: /home/micha/.gitignore
[2024-10-14][14:10:15][ignore::gitignore][DEBUG] opened gitignore file: /home/micha/.gitignore
[2024-10-14][14:10:15][ignore::gitignore][DEBUG] opened gitignore file: /home/micha/astral/test/.git/info/exclude
[2024-10-14][14:10:15][ignore::gitignore][DEBUG] opened gitignore file: /home/micha/astral/test/ruff-ignore-subdir-bug/.gitignore
[2024-10-14][14:10:15][ignore::gitignore][DEBUG] opened gitignore file: /home/micha/astral/test/ruff-ignore-subdir-bug/.git/info/exclude
[2024-10-14][14:10:15][ruff_workspace::resolver][DEBUG] Ignored path via `exclude`: "/home/micha/astral/test/ruff-ignore-subdir-bug/.git"
[2024-10-14][14:10:15][ignore::walk][DEBUG] ignoring /home/micha/astral/test/ruff-ignore-subdir-bug/backend/log: Ignore(IgnoreMatch(Gitignore(Glob { from: Some("/home/micha/astral/test/ruff-ignore-subdir-bug/.gitignore"), original: "backend/log", actual: "backend/log", is_whitelist: false, is_only_dir: false })))
[2024-10-14][14:10:15][ruff_workspace::resolver][DEBUG] Included path via `include`: "/home/micha/astral/test/ruff-ignore-subdir-bug/backend/src/subdir/log/do_not_ignore_me.py"
[2024-10-14][14:10:15][ruff::commands::check][DEBUG] Identified files to lint in: 5.132392ms
[2024-10-14][14:10:15][ruff::diagnostics][DEBUG] Checking: /home/micha/astral/test/ruff-ignore-subdir-bug/backend/src/subdir/log/do_not_ignore_me.py
[2024-10-14][14:10:15][ruff::commands::check][DEBUG] Checked 1 files in: 388.274µs
backend/src/subdir/log/do_not_ignore_me.py:1:1: D100 Missing docstring in public module
Found 1 error.
|
Please try As I mentioned above, the reason we are running ruff only for backend is because we have another pieces of functionality, written in python (behave tests, to be more precise). We are not managing their code and they didn't switch to ruff yet. So calling |
Just wanted to say that I was bit by this as well, in the same manner, reproducible with a small git repo (using ruff 0.8.5): mkdir /tmp/ruff-repro
cd /tmp/ruff-repro
git init
cat <<EOF | git apply -
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..f189725
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1 @@
+web/__init__.py
diff --git a/web/myapp/__init__.py b/web/myapp/__init__.py
new file mode 100644
index 0000000..e69de29
diff --git a/web/myapp/models/__init__.py b/web/myapp/models/__init__.py
new file mode 100644
index 0000000..da5fac8
--- /dev/null
+++ b/web/myapp/models/__init__.py
@@ -0,0 +1 @@
+from foo import *
diff --git a/web/myapp/models/foo.py b/web/myapp/models/foo.py
new file mode 100644
index 0000000..e69de29
EOF Running $ ruff --verbose check .
[2025-01-03][08:29:35][ruff::resolve][DEBUG] Using Ruff default settings
[2025-01-03][08:29:35][ignore::gitignore][DEBUG] opened gitignore file: /home/[email protected]/.config/git/.gitignore
[2025-01-03][08:29:35][ignore::gitignore][DEBUG] opened gitignore file: /tmp/foo/.gitignore
[2025-01-03][08:29:35][ignore::gitignore][DEBUG] opened gitignore file: /tmp/foo/.git/info/exclude
[2025-01-03][08:29:35][ignore::walk][DEBUG] ignoring /tmp/foo/.ruff_cache: Ignore(IgnoreMatch(Gitignore(Glob { from: Some("/home/[email protected]/.config/git/.gitignore"), original: ".ruff_cache", actual: "**/.ruff_cache", is_whitelist: false, is_only_dir: false })))
[2025-01-03][08:29:35][ruff_workspace::resolver][DEBUG] Ignored path via `exclude`: "/tmp/foo/.git"
[2025-01-03][08:29:35][ruff_workspace::resolver][DEBUG] Included path via `include`: "/tmp/foo/web/myapp/__init__.py"
[2025-01-03][08:29:35][ruff_workspace::resolver][DEBUG] Included path via `include`: "/tmp/foo/web/myapp/models/__init__.py"
[2025-01-03][08:29:35][ruff_workspace::resolver][DEBUG] Included path via `include`: "/tmp/foo/web/myapp/models/foo.py"
[2025-01-03][08:29:35][ruff::commands::check][DEBUG] Identified files to lint in: 7.007262ms
[2025-01-03][08:29:35][ruff::commands::check][DEBUG] Checked 3 files in: 195.033µs
web/myapp/models/__init__.py:1:1: F403 `from foo import *` used; unable to detect undefined names
|
1 | from foo import *
| ^^^^^^^^^^^^^^^^^ F403
|
Found 1 error. But $ ruff --verbose check web
[2025-01-03][08:29:37][ruff::resolve][DEBUG] Using Ruff default settings
[2025-01-03][08:29:37][ignore::gitignore][DEBUG] opened gitignore file: /home/[email protected]/.config/git/.gitignore
[2025-01-03][08:29:37][ignore::gitignore][DEBUG] opened gitignore file: /tmp/foo/.gitignore
[2025-01-03][08:29:37][ignore::gitignore][DEBUG] opened gitignore file: /tmp/foo/.git/info/exclude
[2025-01-03][08:29:37][ignore::walk][DEBUG] ignoring /tmp/foo/web/myapp/__init__.py: Ignore(IgnoreMatch(Gitignore(Glob { from: Some("/tmp/foo/.gitignore"), original: "web/__init__.py", actual: "web/__init__.py", is_whitelist: false, is_only_dir: false })))
[2025-01-03][08:29:37][ignore::walk][DEBUG] ignoring /tmp/foo/web/myapp/models/__init__.py: Ignore(IgnoreMatch(Gitignore(Glob { from: Some("/tmp/foo/.gitignore"), original: "web/__init__.py", actual: "web/__init__.py", is_whitelist: false, is_only_dir: false })))
[2025-01-03][08:29:37][ruff_workspace::resolver][DEBUG] Included path via `include`: "/tmp/foo/web/myapp/models/foo.py"
[2025-01-03][08:29:37][ruff::commands::check][DEBUG] Identified files to lint in: 3.593057ms
[2025-01-03][08:29:37][ruff::commands::check][DEBUG] Checked 1 files in: 43.404µs
All checks passed! Even more perplexing to me is that if I repeatedly run However, this only happens in my main repository and I haven't been able to reproduce this inconsistent behavior in the toy repo example above. |
Recently I have faced two issues with ruff. And they might not be that tightly related, but the description for the first one serves as a context for the second one, so I have combined them into a single issue. Let me know if it would be better to separate them.
Here is a minimal reproducible example I was able to prepare.
cd ruff-ignore-subdir-bug git init
.gitignore
file with the following content:Contents of backend/pyproject.toml:
Contents of backend/log/example.py:
Contents of backend/src/subdir/log/some_logging_lib.py:
1. Issue with ignore rule and subdir
Ruff mistakenly ignores backend/src/subdir/log, while it shouldn't.
Actual behavior:
Expected behavior:
Here are two log entries I was able to see when added
--verbose
flag:Also, I might be not that familiar with rust, but I checked Cargo.toml file, and I was able to find that you are using ignore crate, which is a part of ripgrep repository. So this issue might be related: BurntSushi/ripgrep#2778
2. Ruff doesn't apply
respect-gitignore = false
when pyproject.toml is not in the same directory where you run commandI can't tell too much details about the real project where I faced this issue, but we have our pyproject.toml located inside of backend directory and not in the root, since there are other directories and pieces of functionality, which also might contain python code, and they are not using ruff linter. So running linter from the root of repository without arguments will produce a lot of errors. That's why I run commands like
ruff check backend
orcd backend && ruff check
, but notruff check
from root.So once I have faced the issue above with .gitignore and sub-directories, I have added the following to the backend/pyproject.toml file:
And here is what it shows.
Actual behavior:
Expected behavior:
🤷♂️ 😄
I don't know whether it's a correct expectation, but since ruff already applies lint rules to backend/ and it's sub-directories, listed in backend/pyproject.toml, it feels logical that it should apply this
respect-gitignore
setting as well.We don't have a configured git hooks via pre-commit, or something similar, so I have added the following pre-commit hook
ruff check backend
for myself, since ruff is fast enough to do that (thank you so much for that, btw). And once I have found that there might be issues with ignoring sub-directories, I changed the command to beruff check --no-respect-gitignore backend
.The text was updated successfully, but these errors were encountered: