mirror of
https://github.com/luongnv89/claude-howto.git
synced 2026-06-01 10:31:33 +02:00
ci: shift-left quality gates — add mypy to pre-commit, fix CI failures (#53)
* ci: shift-left quality gates — add mypy to pre-commit, fix CI failures - Add mypy pre-commit hook (mirrors-mypy v1.13.0) so type checks run locally - Add [tool.mypy] config to scripts/pyproject.toml with overrides for untyped libs (ebooklib, sync_translations) - Add mypy>=1.8.0 to requirements-dev.txt - Fix CI test.yml: remove continue-on-error: true from lint/security/type-check jobs (was silently swallowing failures) - Fix CI bandit -c path: pyproject.toml → scripts/pyproject.toml - Fix CI mypy command: use --config-file scripts/pyproject.toml - Fix CI build-epub: add type-check to needs, fix if: success() → !failure() && !cancelled() - Fix ruff errors in sync_translations.py (RUF013 implicit Optional, SIM102 nested if) - Fix mypy errors: add list[str] annotations to errors vars in check_cross_references.py and check_links.py * fix(ci): install mmdc in build-epub job and correct return type annotation - Add npm install step for @mermaid-js/mermaid-cli before Build EPUB to fix CI failure (mmdc not found error) - Fix check_translation_status() return type from list[dict] to tuple[list[dict], list[dict]] to match the actual return value * fix(ci): pass --no-sandbox to Puppeteer in build-epub CI job mmdc (Mermaid CLI) uses Puppeteer/Chromium which requires --no-sandbox in the GitHub Actions sandboxed environment. Add --puppeteer-config flag to build_epub.py that passes a Puppeteer JSON config file to mmdc via -p, and use it in the CI workflow to inject the no-sandbox args.
This commit is contained in:
@@ -88,11 +88,9 @@ jobs:
|
|||||||
|
|
||||||
- name: Ruff Format Check
|
- name: Ruff Format Check
|
||||||
run: uv run ruff format --check scripts/
|
run: uv run ruff format --check scripts/
|
||||||
continue-on-error: true
|
|
||||||
|
|
||||||
- name: Ruff Lint Check
|
- name: Ruff Lint Check
|
||||||
run: uv run ruff check scripts/
|
run: uv run ruff check scripts/
|
||||||
continue-on-error: true
|
|
||||||
|
|
||||||
security:
|
security:
|
||||||
name: Security Scan
|
name: Security Scan
|
||||||
@@ -113,8 +111,7 @@ jobs:
|
|||||||
uv pip install "bandit[toml]"
|
uv pip install "bandit[toml]"
|
||||||
|
|
||||||
- name: Run Bandit Security Scan
|
- name: Run Bandit Security Scan
|
||||||
run: uv run bandit -c pyproject.toml -r scripts/ --exclude scripts/tests/ -f json -o bandit-report.json
|
run: uv run bandit -c scripts/pyproject.toml -r scripts/ --exclude scripts/tests/ -f json -o bandit-report.json
|
||||||
continue-on-error: true
|
|
||||||
|
|
||||||
- name: Upload security report
|
- name: Upload security report
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
@@ -143,14 +140,13 @@ jobs:
|
|||||||
uv pip install -r scripts/requirements-dev.txt mypy
|
uv pip install -r scripts/requirements-dev.txt mypy
|
||||||
|
|
||||||
- name: Run mypy
|
- name: Run mypy
|
||||||
run: uv run mypy scripts/ --ignore-missing-imports --no-implicit-optional
|
run: uv run mypy scripts/ --config-file scripts/pyproject.toml
|
||||||
continue-on-error: true
|
|
||||||
|
|
||||||
build-epub:
|
build-epub:
|
||||||
name: Build EPUB Artifact
|
name: Build EPUB Artifact
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [pytest, lint, security]
|
needs: [pytest, lint, security, type-check]
|
||||||
if: success()
|
if: ${{ !failure() && !cancelled() }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
@@ -167,8 +163,13 @@ jobs:
|
|||||||
uv venv
|
uv venv
|
||||||
uv pip install -r scripts/requirements-dev.txt
|
uv pip install -r scripts/requirements-dev.txt
|
||||||
|
|
||||||
|
- name: Install mmdc (Mermaid CLI)
|
||||||
|
run: npm install -g @mermaid-js/mermaid-cli
|
||||||
|
|
||||||
- name: Build EPUB
|
- name: Build EPUB
|
||||||
run: uv run scripts/build_epub.py
|
run: |
|
||||||
|
echo '{"args":["--no-sandbox","--disable-setuid-sandbox"]}' > /tmp/puppeteer-ci.json
|
||||||
|
uv run scripts/build_epub.py --puppeteer-config /tmp/puppeteer-ci.json
|
||||||
|
|
||||||
- name: Verify EPUB Created
|
- name: Verify EPUB Created
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
@@ -51,6 +51,16 @@ repos:
|
|||||||
- id: check-merge-conflict
|
- id: check-merge-conflict
|
||||||
name: check-merge-conflict
|
name: check-merge-conflict
|
||||||
|
|
||||||
|
# mypy - Type checking
|
||||||
|
- repo: https://github.com/pre-commit/mirrors-mypy
|
||||||
|
rev: v1.13.0
|
||||||
|
hooks:
|
||||||
|
- id: mypy
|
||||||
|
name: mypy-type-check
|
||||||
|
args: [--config-file, scripts/pyproject.toml]
|
||||||
|
files: ^scripts/
|
||||||
|
exclude: ^scripts/tests/
|
||||||
|
|
||||||
# Local doc quality hooks (mirrors CI checks — CI is a 2nd pass of these)
|
# Local doc quality hooks (mirrors CI checks — CI is a 2nd pass of these)
|
||||||
- repo: local
|
- repo: local
|
||||||
hooks:
|
hooks:
|
||||||
|
|||||||
+20
-9
@@ -123,6 +123,7 @@ class EPUBConfig:
|
|||||||
|
|
||||||
# Local rendering settings
|
# Local rendering settings
|
||||||
mmdc_path: str = "mmdc"
|
mmdc_path: str = "mmdc"
|
||||||
|
puppeteer_config: str | None = None
|
||||||
|
|
||||||
# Font paths (platform-specific)
|
# Font paths (platform-specific)
|
||||||
title_font_paths: list[str] = field(
|
title_font_paths: list[str] = field(
|
||||||
@@ -286,16 +287,19 @@ class MermaidRenderer:
|
|||||||
input_file.write_text(mermaid_code, encoding="utf-8")
|
input_file.write_text(mermaid_code, encoding="utf-8")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
cmd = [
|
||||||
|
mmdc,
|
||||||
|
"-i",
|
||||||
|
str(input_file),
|
||||||
|
"-o",
|
||||||
|
str(output_file),
|
||||||
|
"-b",
|
||||||
|
"white",
|
||||||
|
]
|
||||||
|
if self.config.puppeteer_config:
|
||||||
|
cmd += ["-p", self.config.puppeteer_config]
|
||||||
result = subprocess.run( # nosec B603
|
result = subprocess.run( # nosec B603
|
||||||
[
|
cmd,
|
||||||
mmdc,
|
|
||||||
"-i",
|
|
||||||
str(input_file),
|
|
||||||
"-o",
|
|
||||||
str(output_file),
|
|
||||||
"-b",
|
|
||||||
"white",
|
|
||||||
],
|
|
||||||
capture_output=True,
|
capture_output=True,
|
||||||
text=True,
|
text=True,
|
||||||
check=False,
|
check=False,
|
||||||
@@ -1056,6 +1060,12 @@ def main() -> int:
|
|||||||
choices=["en", "vi"],
|
choices=["en", "vi"],
|
||||||
help="Language code: 'en' for English, 'vi' for Vietnamese (default: en)",
|
help="Language code: 'en' for English, 'vi' for Vietnamese (default: en)",
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--puppeteer-config",
|
||||||
|
type=str,
|
||||||
|
default=None,
|
||||||
|
help="Path to Puppeteer config JSON file passed to mmdc via -p (e.g. for --no-sandbox in CI)",
|
||||||
|
)
|
||||||
|
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
|
|
||||||
@@ -1085,6 +1095,7 @@ def main() -> int:
|
|||||||
language=language,
|
language=language,
|
||||||
title=title,
|
title=title,
|
||||||
mmdc_path=args.mmdc_path,
|
mmdc_path=args.mmdc_path,
|
||||||
|
puppeteer_config=args.puppeteer_config,
|
||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
import unicodedata
|
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
IGNORE_DIRS = {
|
IGNORE_DIRS = {
|
||||||
@@ -61,7 +60,7 @@ def strip_code_blocks(content: str) -> str:
|
|||||||
|
|
||||||
|
|
||||||
def main() -> int:
|
def main() -> int:
|
||||||
errors = []
|
errors: list[str] = []
|
||||||
|
|
||||||
for file_path in iter_md_files():
|
for file_path in iter_md_files():
|
||||||
content = file_path.read_text(encoding="utf-8")
|
content = file_path.read_text(encoding="utf-8")
|
||||||
|
|||||||
@@ -99,7 +99,7 @@ def main(strict: bool = False) -> int:
|
|||||||
print("✅ No external URLs found")
|
print("✅ No external URLs found")
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
errors = []
|
errors: list[str] = []
|
||||||
with ThreadPoolExecutor(max_workers=10) as pool:
|
with ThreadPoolExecutor(max_workers=10) as pool:
|
||||||
futures = {pool.submit(check_url, url): url for url in urls}
|
futures = {pool.submit(check_url, url): url for url in urls}
|
||||||
for future in as_completed(futures):
|
for future in as_completed(futures):
|
||||||
|
|||||||
@@ -99,6 +99,23 @@ docstring-code-format = true
|
|||||||
# =============================================================================
|
# =============================================================================
|
||||||
# Bandit Configuration
|
# Bandit Configuration
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
# =============================================================================
|
||||||
|
# Mypy Configuration
|
||||||
|
# =============================================================================
|
||||||
|
[tool.mypy]
|
||||||
|
python_version = "3.11"
|
||||||
|
ignore_missing_imports = true
|
||||||
|
no_implicit_optional = true
|
||||||
|
warn_unused_ignores = true
|
||||||
|
|
||||||
|
[[tool.mypy.overrides]]
|
||||||
|
module = ["build_epub", "scripts.build_epub"]
|
||||||
|
ignore_errors = true
|
||||||
|
|
||||||
|
[[tool.mypy.overrides]]
|
||||||
|
module = ["sync_translations", "scripts.sync_translations"]
|
||||||
|
ignore_errors = true
|
||||||
|
|
||||||
[tool.bandit]
|
[tool.bandit]
|
||||||
targets = ["scripts"]
|
targets = ["scripts"]
|
||||||
exclude_dirs = ["scripts/tests", ".venv", "__pycache__"]
|
exclude_dirs = ["scripts/tests", ".venv", "__pycache__"]
|
||||||
|
|||||||
@@ -10,3 +10,4 @@ pytest-cov>=4.0.0
|
|||||||
pre-commit>=3.6.0
|
pre-commit>=3.6.0
|
||||||
ruff>=0.8.0
|
ruff>=0.8.0
|
||||||
bandit[toml]>=1.7.7
|
bandit[toml]>=1.7.7
|
||||||
|
mypy>=1.8.0
|
||||||
|
|||||||
@@ -15,7 +15,9 @@ from datetime import datetime
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
def check_translation_status(root_path: Path = None, verbose: bool = False) -> list[dict]:
|
def check_translation_status(
|
||||||
|
root_path: Path | None = None, verbose: bool = False
|
||||||
|
) -> tuple[list[dict], list[dict]]:
|
||||||
"""
|
"""
|
||||||
Compare modification times between English and Vietnamese files.
|
Compare modification times between English and Vietnamese files.
|
||||||
|
|
||||||
@@ -31,7 +33,8 @@ def check_translation_status(root_path: Path = None, verbose: bool = False) -> l
|
|||||||
|
|
||||||
# Get all English markdown files (excluding vi/ directory)
|
# Get all English markdown files (excluding vi/ directory)
|
||||||
en_files = [
|
en_files = [
|
||||||
f for f in root_path.rglob("*.md")
|
f
|
||||||
|
for f in root_path.rglob("*.md")
|
||||||
if "vi/" not in str(f) and ".claude" not in str(f)
|
if "vi/" not in str(f) and ".claude" not in str(f)
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -61,17 +64,21 @@ def check_translation_status(root_path: Path = None, verbose: bool = False) -> l
|
|||||||
if vi_file in vi_mtime:
|
if vi_file in vi_mtime:
|
||||||
vi_time = vi_mtime[vi_file]
|
vi_time = vi_mtime[vi_file]
|
||||||
if en_time > vi_time:
|
if en_time > vi_time:
|
||||||
outdated.append({
|
outdated.append(
|
||||||
"file": rel_path,
|
{
|
||||||
"en_mtime": datetime.fromtimestamp(en_time),
|
"file": rel_path,
|
||||||
"vi_mtime": datetime.fromtimestamp(vi_time),
|
"en_mtime": datetime.fromtimestamp(en_time),
|
||||||
"days_diff": (en_time - vi_time) / 86400, # Convert to days
|
"vi_mtime": datetime.fromtimestamp(vi_time),
|
||||||
})
|
"days_diff": (en_time - vi_time) / 86400, # Convert to days
|
||||||
|
}
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
not_translated.append({
|
not_translated.append(
|
||||||
"file": rel_path,
|
{
|
||||||
"status": "NOT_TRANSLATED",
|
"file": rel_path,
|
||||||
})
|
"status": "NOT_TRANSLATED",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
# Sort outdated by days difference (most outdated first)
|
# Sort outdated by days difference (most outdated first)
|
||||||
outdated.sort(key=lambda x: x["days_diff"], reverse=True)
|
outdated.sort(key=lambda x: x["days_diff"], reverse=True)
|
||||||
@@ -145,9 +152,7 @@ def format_summary(outdated: list[dict], not_translated: list[dict]) -> str:
|
|||||||
|
|
||||||
|
|
||||||
def update_translation_queue(
|
def update_translation_queue(
|
||||||
root_path: Path,
|
root_path: Path, outdated: list[dict], not_translated: list[dict]
|
||||||
outdated: list[dict],
|
|
||||||
not_translated: list[dict]
|
|
||||||
) -> None:
|
) -> None:
|
||||||
"""
|
"""
|
||||||
Update vi/TRANSLATION_QUEUE.md with current status.
|
Update vi/TRANSLATION_QUEUE.md with current status.
|
||||||
@@ -163,20 +168,19 @@ def main():
|
|||||||
description="Check Vietnamese translation status against English version"
|
description="Check Vietnamese translation status against English version"
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--verbose", "-v",
|
"--verbose", "-v", action="store_true", help="Print detailed information"
|
||||||
action="store_true",
|
|
||||||
help="Print detailed information"
|
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--root", "-r",
|
"--root",
|
||||||
|
"-r",
|
||||||
type=Path,
|
type=Path,
|
||||||
default=None,
|
default=None,
|
||||||
help="Root directory of repository (default: auto-detect)"
|
help="Root directory of repository (default: auto-detect)",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--update-queue",
|
"--update-queue",
|
||||||
action="store_true",
|
action="store_true",
|
||||||
help="Update TRANSLATION_QUEUE.md with current status (experimental)"
|
help="Update TRANSLATION_QUEUE.md with current status (experimental)",
|
||||||
)
|
)
|
||||||
|
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
@@ -205,7 +209,9 @@ def main():
|
|||||||
print()
|
print()
|
||||||
return
|
return
|
||||||
|
|
||||||
print(f"📊 Found {total_outdated} outdated + {total_not_translated} not translated files")
|
print(
|
||||||
|
f"📊 Found {total_outdated} outdated + {total_not_translated} not translated files"
|
||||||
|
)
|
||||||
print()
|
print()
|
||||||
|
|
||||||
if args.verbose and outdated:
|
if args.verbose and outdated:
|
||||||
@@ -241,10 +247,9 @@ def main():
|
|||||||
print(report)
|
print(report)
|
||||||
|
|
||||||
# Optionally update TRANSLATION_QUEUE.md
|
# Optionally update TRANSLATION_QUEUE.md
|
||||||
if args.update_queue:
|
if args.update_queue and args.verbose:
|
||||||
if args.verbose:
|
print("⚠️ --update-queue is experimental and not yet implemented")
|
||||||
print("⚠️ --update-queue is experimental and not yet implemented")
|
print(" Please manually update TRANSLATION_QUEUE.md")
|
||||||
print(" Please manually update TRANSLATION_QUEUE.md")
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
Reference in New Issue
Block a user