initial commit
This commit is contained in:
commit
86456cafa3
133
.gitignore
vendored
Normal file
133
.gitignore
vendored
Normal file
@ -0,0 +1,133 @@
|
||||
# Byte-compiled / optimized / DLL files
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
|
||||
# Distribution / packaging
|
||||
.Python
|
||||
build/
|
||||
develop-eggs/
|
||||
dist/
|
||||
downloads/
|
||||
eggs/
|
||||
.eggs/
|
||||
lib/
|
||||
lib64/
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
wheels/
|
||||
share/python-wheels/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
MANIFEST
|
||||
|
||||
# PyInstaller
|
||||
# Usually these files are written by a python script from a template
|
||||
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||
*.manifest
|
||||
*.spec
|
||||
|
||||
# Installer logs
|
||||
pip-log.txt
|
||||
pip-delete-this-directory.txt
|
||||
|
||||
# Unit test / coverage reports
|
||||
htmlcov/
|
||||
.tox/
|
||||
.nox/
|
||||
.coverage
|
||||
.coverage.*
|
||||
.cache
|
||||
nosetests.xml
|
||||
coverage.xml
|
||||
*.cover
|
||||
*.py,cover
|
||||
.hypothesis/
|
||||
.pytest_cache/
|
||||
cover/
|
||||
|
||||
# Sphinx documentation
|
||||
docs/_build/
|
||||
|
||||
# PyBuilder
|
||||
.pybuilder/
|
||||
target/
|
||||
|
||||
# Jupyter Notebook
|
||||
.ipynb_checkpoints
|
||||
|
||||
# IPython
|
||||
profile_default/
|
||||
ipython_config.py
|
||||
|
||||
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
||||
__pypackages__/
|
||||
|
||||
# Celery stuff
|
||||
celerybeat-schedule
|
||||
celerybeat.pid
|
||||
|
||||
# SageMath parsed files
|
||||
*.sage.py
|
||||
|
||||
# Environments
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
.venv
|
||||
env/
|
||||
venv/
|
||||
ENV/
|
||||
env.bak/
|
||||
venv.bak/
|
||||
|
||||
# Spyder project settings
|
||||
.spyderproject
|
||||
.spyproject
|
||||
|
||||
# Rope project settings
|
||||
.ropeproject
|
||||
|
||||
# mkdocs documentation
|
||||
/site
|
||||
|
||||
# mypy
|
||||
.mypy_cache/
|
||||
.dmypy.json
|
||||
dmypy.json
|
||||
|
||||
# Pyre type checker
|
||||
.pyre/
|
||||
|
||||
# pytype static type analyzer
|
||||
.pytype/
|
||||
|
||||
# Cython debug symbols
|
||||
cython_debug/
|
||||
|
||||
# PyCharm
|
||||
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||
#.idea/
|
||||
|
||||
# Ruff stuff:
|
||||
.ruff_cache/
|
||||
|
||||
# PyPI configuration file
|
||||
.pypirc
|
||||
|
||||
# Custom:
|
||||
logs/
|
||||
log/
|
||||
release/
|
||||
#media/
|
||||
*.zip
|
||||
localstorage/
|
||||
.vscode/
|
||||
*.txt
|
||||
*.ini
|
||||
TODO.md
|
18
CHANGELOG.md
Normal file
18
CHANGELOG.md
Normal file
@ -0,0 +1,18 @@
|
||||
# CHANGELOG
|
||||
|
||||
- 2025-04-25 - initial commit
|
||||
|
||||
- **Geändert:**
|
||||
- [x] `treescanner.py` umgebaut zu kombinierter Modul- und Standalone-Version
|
||||
- [x] Konfigurationsklasse `TreeScannerConfig` eingebaut
|
||||
- [x] Klasse `TreeScanner` erstellt und bestehende Logik dorthin verschoben
|
||||
|
||||
- **Hinzugefügt:**
|
||||
- [x] `test_usage.py` als __Beispiel__ für modulare Verwendung
|
||||
- [x] `README.md` erstellt mit Anleitung für Standalone- und Modulnutzung (Template)
|
||||
- [x] `pyproject.toml` erstellt für spätere Paketinstallation mit PEP 621
|
||||
- [ ] `TODO.md` ist eingefügt aber wird vorerst noch nicht versioniert.
|
||||
|
||||
- **Geprüft:**
|
||||
- [x] `scanner.py` solo in ein Verzeichnis kopieren und ausführen mit `python scanner.py`
|
||||
- [x] Import und Nutzung als Modul aus `test_usage.py`
|
23
LICENSE
Normal file
23
LICENSE
Normal file
@ -0,0 +1,23 @@
|
||||
MIT License with Attribution Requirement
|
||||
|
||||
Copyright (c) 2025 Adam Skotarczak <adam@skotarczak.net>
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the “Software”), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, subject to the following conditions:
|
||||
|
||||
1. The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
2. **Attribution Requirement**: Any public use or distribution of this Software,
|
||||
modified or unmodified, must include a clear and visible attribution to the original author:
|
||||
|
||||
**Adam Skotarczak <adam@skotarczak.net>**
|
||||
|
||||
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
||||
INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
||||
PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
|
||||
FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
48
README.md
Normal file
48
README.md
Normal file
@ -0,0 +1,48 @@
|
||||
# TreeScanner
|
||||
|
||||
Ein flexibler Verzeichnisscanner für die Kommandozeile **und** zur Einbindung als Python-Modul.
|
||||
|
||||
## Projektstruktur
|
||||
|
||||
```plaintext
|
||||
|
||||
treescanner/
|
||||
├── __init__.py
|
||||
├── __main__.py # Standalone-Ausführung
|
||||
├── scanner.py # Das eigentliche Modul mit Klasse + Logik
|
||||
├── config.py # Konfigurationsklasse separat
|
||||
└── test_usage.py # Beispielverwendung als Modul
|
||||
```
|
||||
|
||||
## 🔧 Verwendung als Standalone
|
||||
|
||||
```bash
|
||||
python treescanner.py
|
||||
```
|
||||
|
||||
Erzeugt eine Datei `tree.txt` mit der Verzeichnisstruktur ab dem aktuellen Pfad.
|
||||
|
||||
## 🧩 Verwendung als Modul
|
||||
|
||||
```python
|
||||
from treescanner import TreeScanner, TreeScannerConfig
|
||||
|
||||
config = TreeScannerConfig(root_path=".", max_depth=2)
|
||||
scanner = TreeScanner(config)
|
||||
output = scanner.generate_tree()
|
||||
print(output)
|
||||
```
|
||||
|
||||
## ⚙️ Konfiguration
|
||||
|
||||
Die `TreeScannerConfig`-Klasse erlaubt dir u. a.:
|
||||
|
||||
- `root_path`: Startverzeichnis
|
||||
- `max_depth`: maximale Rekursionstiefe
|
||||
- `max_files_per_dir`: wie viele Dateien pro Ordner angezeigt werden
|
||||
- `align_comments`: Kommentar-Ausrichtung aktivieren
|
||||
- `folder_icon` / `file_icon`: Anzeige-Icons
|
||||
|
||||
## 📄 Lizenz
|
||||
|
||||
MIT (optional anpassen)
|
0
__init__.py
Normal file
0
__init__.py
Normal file
0
__main__.py
Normal file
0
__main__.py
Normal file
21
config.py
Normal file
21
config.py
Normal file
@ -0,0 +1,21 @@
|
||||
|
||||
# config.py
|
||||
# Beispielconfig:
|
||||
#
|
||||
# from .config import TreeScannerConfig
|
||||
#
|
||||
|
||||
class TreeScannerConfig:
|
||||
def __init__(self,
|
||||
root_path=".",
|
||||
folder_icon="📁",
|
||||
file_icon="📄",
|
||||
max_files_per_dir=2,
|
||||
max_depth=None,
|
||||
align_comments=True):
|
||||
self.root_path = root_path
|
||||
self.folder_icon = folder_icon
|
||||
self.file_icon = file_icon
|
||||
self.max_files_per_dir = max_files_per_dir
|
||||
self.max_depth = max_depth
|
||||
self.align_comments = align_comments
|
BIN
media/favicon.ico
Normal file
BIN
media/favicon.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 74 KiB |
BIN
media/logo-bw-1024x1024.png
Normal file
BIN
media/logo-bw-1024x1024.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.4 MiB |
BIN
media/logo-bw-alpha-512x512.png
Normal file
BIN
media/logo-bw-alpha-512x512.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 143 KiB |
BIN
media/logo-colour-1024x1024.png
Normal file
BIN
media/logo-colour-1024x1024.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.4 MiB |
BIN
media/logo-colour-alpha-512x512.png
Normal file
BIN
media/logo-colour-alpha-512x512.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 155 KiB |
15
pyproject.toml
Normal file
15
pyproject.toml
Normal file
@ -0,0 +1,15 @@
|
||||
[project]
|
||||
name = "treescanner"
|
||||
version = "0.1.0"
|
||||
description = "Ein Verzeichnisscanner als CLI-Tool und Python-Modul"
|
||||
authors = [
|
||||
{ name="Adam Skotarczak", email="adam@skotarczak.net" }
|
||||
]
|
||||
readme = "README.md"
|
||||
requires-python = ">=3.7"
|
||||
license = { text = "MIT" }
|
||||
keywords = ["filesystem", "tree", "scanner", "cli", "modul"]
|
||||
|
||||
[build-system]
|
||||
requires = ["setuptools"]
|
||||
build-backend = "setuptools.build_meta"
|
88
scanner.py
Normal file
88
scanner.py
Normal file
@ -0,0 +1,88 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
"""
|
||||
Verzeichnisscanner mit strukturierter Ausgabe.
|
||||
Funktioniert sowohl als Standalone-Skript als auch als einbindbares Modul.
|
||||
"""
|
||||
|
||||
import os
|
||||
from typing import Optional, List
|
||||
|
||||
# === Konfigurationsklasse ===
|
||||
class TreeScannerConfig:
|
||||
def __init__(self,
|
||||
root_path: str = ".",
|
||||
folder_icon: str = "\U0001F4C1",
|
||||
file_icon: str = "\U0001F4C4",
|
||||
max_files_per_dir: int = 100,
|
||||
max_depth: Optional[int] = None,
|
||||
align_comments: bool = True): #
|
||||
self.root_path = root_path
|
||||
self.folder_icon = folder_icon
|
||||
self.file_icon = file_icon
|
||||
self.max_files_per_dir = max_files_per_dir
|
||||
self.max_depth = max_depth
|
||||
self.align_comments = align_comments
|
||||
|
||||
# === Hauptklasse ===
|
||||
class TreeScanner:
|
||||
def __init__(self, config: TreeScannerConfig):
|
||||
self.config = config
|
||||
|
||||
def scan_directory(self, path: str, depth: int = 0, prefix: str = "") -> List[str]:
|
||||
lines = []
|
||||
try:
|
||||
entries = sorted(os.listdir(path))
|
||||
except PermissionError:
|
||||
return [f"{prefix}└── [Zugriff verweigert] {path}"]
|
||||
|
||||
folders = [e for e in entries if os.path.isdir(os.path.join(path, e))]
|
||||
files = [e for e in entries if os.path.isfile(os.path.join(path, e))]
|
||||
|
||||
for idx, folder in enumerate(folders):
|
||||
folder_path = os.path.join(path, folder)
|
||||
connector = "├── " if idx < len(folders) - 1 or files else "└── "
|
||||
line = f"{prefix}{connector}{self.config.folder_icon} {folder}"
|
||||
lines.append(line)
|
||||
if self.config.max_depth is None or depth < self.config.max_depth:
|
||||
extension = "│ " if idx < len(folders) - 1 or files else " "
|
||||
lines.extend(self.scan_directory(folder_path, depth + 1, prefix + extension))
|
||||
|
||||
for idx, file in enumerate(files[:self.config.max_files_per_dir]):
|
||||
connector = "├── " if idx < len(files[:self.config.max_files_per_dir]) - 1 else "└── "
|
||||
line = f"{prefix}{connector}{self.config.file_icon} {file}"
|
||||
lines.append(line)
|
||||
|
||||
if len(files) > self.config.max_files_per_dir:
|
||||
remaining = len(files) - self.config.max_files_per_dir
|
||||
lines.append(f"{prefix}└── <und {remaining} weitere Dateien>")
|
||||
|
||||
return lines
|
||||
|
||||
def align_lines_with_comments(self, lines: List[str]) -> List[str]:
|
||||
max_length = max(len(line.rstrip()) for line in lines)
|
||||
return [
|
||||
line.rstrip() + (" " * (max_length - len(line.rstrip()) + 2)) + "# "
|
||||
for line in lines
|
||||
]
|
||||
|
||||
def generate_tree(self) -> str:
|
||||
root_name = os.path.basename(os.path.abspath(self.config.root_path)) or self.config.root_path
|
||||
lines = [f"{self.config.folder_icon} {root_name}/"]
|
||||
lines += self.scan_directory(self.config.root_path)
|
||||
|
||||
if self.config.align_comments:
|
||||
lines = self.align_lines_with_comments(lines)
|
||||
|
||||
return "\n".join(lines)
|
||||
|
||||
# === Standalone-Ausführung ===
|
||||
def main():
|
||||
config = TreeScannerConfig()
|
||||
scanner = TreeScanner(config)
|
||||
tree_output = scanner.generate_tree()
|
||||
with open("tree.txt", "w", encoding="utf-8") as f:
|
||||
f.write(tree_output + "\n")
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
14
test_usage.py
Normal file
14
test_usage.py
Normal file
@ -0,0 +1,14 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from scanner import TreeScanner, TreeScannerConfig
|
||||
|
||||
# Beispielkonfiguration
|
||||
config = TreeScannerConfig(
|
||||
root_path=".",
|
||||
max_depth=2,
|
||||
align_comments=True
|
||||
)
|
||||
|
||||
scanner = TreeScanner(config)
|
||||
baum = scanner.generate_tree()
|
||||
print(baum)
|
Loading…
x
Reference in New Issue
Block a user