-
Notifications
You must be signed in to change notification settings - Fork 25
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement pre-commit, pyproject, ruff, mypy
- Loading branch information
Showing
5 changed files
with
351 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,24 @@ | ||
/bin | ||
/lib | ||
/pyvenv.cfg | ||
*.pyc | ||
.vscode | ||
.DS_Store | ||
Thumbs.db | ||
[Dd]esktop.ini | ||
$RECYCLE.BIN/ | ||
*.lnk | ||
|
||
# Environments | ||
.env | ||
.venv | ||
env/ | ||
venv/ | ||
ENV/ | ||
env.bak/ | ||
venv.bak/ | ||
.tox | ||
custom_components/*/__pycache__/ | ||
*.pyc | ||
.coverage | ||
coverage.xml |
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
# See https://pre-commit.com for more information | ||
# See https://pre-commit.com/hooks.html for more hooks | ||
repos: | ||
- repo: https://github.com/pre-commit/pre-commit-hooks | ||
rev: v5.0.0 | ||
hooks: | ||
- id: trailing-whitespace | ||
exclude: ".*.md" | ||
- id: end-of-file-fixer | ||
exclude: ".*.md" | ||
- id: check-yaml | ||
- id: check-toml | ||
- id: check-added-large-files | ||
- repo: https://github.com/rhysd/actionlint | ||
rev: v1.7.4 | ||
hooks: | ||
- id: actionlint | ||
- repo: https://github.com/codespell-project/codespell | ||
rev: v2.2.4 | ||
hooks: | ||
- id: codespell | ||
additional_dependencies: | ||
- tomli | ||
- repo: https://github.com/astral-sh/ruff-pre-commit | ||
rev: v0.8.3 | ||
hooks: | ||
# Run the linter. | ||
- id: ruff | ||
args: [--fix] | ||
# Run the formatter. | ||
- id: ruff-format |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,289 @@ | ||
[tool.codespell] | ||
skip = "*.py,*.toml,CHANGELOG.*" | ||
quiet-level = 3 | ||
ignore-words-list = "hass" | ||
|
||
[tool.coverage.report] | ||
show_missing = true | ||
fail_under = 80 | ||
|
||
[tool.mypy] | ||
python_version = "3.13" | ||
follow_imports = "silent" | ||
ignore_missing_imports = true | ||
platform = "linux" | ||
show_error_codes = true | ||
local_partial_types = true | ||
strict_equality = true | ||
no_implicit_optional = true | ||
warn_incomplete_stub = true | ||
warn_redundant_casts = true | ||
warn_unused_configs = true | ||
warn_unused_ignores = true | ||
enable_error_code = ["deprecated", "ignore-without-code", "redundant-self", "truthy-iterable"] | ||
disable_error_code = ["annotation-unchecked", "import-not-found", "import-untyped", "call-arg"] | ||
extra_checks = false | ||
check_untyped_defs = true | ||
disallow_incomplete_defs = false | ||
disallow_subclassing_any = false | ||
disallow_untyped_calls = false | ||
disallow_untyped_decorators = false | ||
disallow_untyped_defs = false | ||
warn_return_any = false | ||
warn_unreachable = false | ||
|
||
[tool.pytest.ini_options] | ||
testpaths = [ | ||
"tests", | ||
] | ||
norecursedirs = [ | ||
".git", | ||
"testing_config", | ||
] | ||
log_format = "%(asctime)s.%(msecs)03d %(levelname)-8s %(threadName)s %(name)s:%(filename)s:%(lineno)s %(message)s" | ||
log_date_format = "%Y-%m-%d %H:%M:%S" | ||
# log_cli = true # Enable live logging to the console | ||
# log_level = "DEBUG" # Set the logging level to DEBUG | ||
asyncio_mode = "auto" | ||
asyncio_default_fixture_loop_scope = "function" | ||
timeout = 30 | ||
addopts = "-rA" | ||
|
||
[tool.ruff] | ||
line-length = 100 | ||
indent-width = 4 | ||
fix = true | ||
force-exclude = true | ||
target-version = "py313" | ||
required-version = ">=0.8.0" | ||
|
||
[tool.ruff.format] | ||
quote-style = "double" | ||
indent-style = "space" | ||
skip-magic-trailing-comma = false | ||
line-ending = "lf" | ||
docstring-code-format = true | ||
docstring-code-line-length = "dynamic" | ||
|
||
[tool.ruff.lint] | ||
select = [ | ||
"A001", # Variable {name} is shadowing a Python builtin | ||
"ASYNC210", # Async functions should not call blocking HTTP methods | ||
"ASYNC220", # Async functions should not create subprocesses with blocking methods | ||
"ASYNC221", # Async functions should not run processes with blocking methods | ||
"ASYNC222", # Async functions should not wait on processes with blocking methods | ||
"ASYNC230", # Async functions should not open files with blocking methods like open | ||
"ASYNC251", # Async functions should not call time.sleep | ||
"B002", # Python does not support the unary prefix increment | ||
"B005", # Using .strip() with multi-character strings is misleading | ||
"B007", # Loop control variable {name} not used within loop body | ||
"B014", # Exception handler with duplicate exception | ||
"B015", # Pointless comparison. Did you mean to assign a value? Otherwise, prepend assert or remove it. | ||
"B017", # pytest.raises(BaseException) should be considered evil | ||
"B018", # Found useless attribute access. Either assign it to a variable or remove it. | ||
"B023", # Function definition does not bind loop variable {name} | ||
"B026", # Star-arg unpacking after a keyword argument is strongly discouraged | ||
"B032", # Possible unintentional type annotation (using :). Did you mean to assign (using =)? | ||
"B904", # Use raise from to specify exception cause | ||
"B905", # zip() without an explicit strict= parameter | ||
"BLE", | ||
"C", # complexity | ||
"COM818", # Trailing comma on bare tuple prohibited | ||
"D", # docstrings | ||
"DTZ003", # Use datetime.now(tz=) instead of datetime.utcnow() | ||
"DTZ004", # Use datetime.fromtimestamp(ts, tz=) instead of datetime.utcfromtimestamp(ts) | ||
"E", # pycodestyle | ||
"F", # pyflakes/autoflake | ||
"F541", # f-string without any placeholders | ||
"FLY", # flynt | ||
"FURB", # refurb | ||
"G", # flake8-logging-format | ||
"I", # isort | ||
"INP", # flake8-no-pep420 | ||
"ISC", # flake8-implicit-str-concat | ||
"ICN001", # import concentions; {name} should be imported as {asname} | ||
"LOG", # flake8-logging | ||
"N804", # First argument of a class method should be named cls | ||
"N805", # First argument of a method should be named self | ||
"N815", # Variable {name} in class scope should not be mixedCase | ||
"PERF", # Perflint | ||
"PGH", # pygrep-hooks | ||
"PIE", # flake8-pie | ||
"PL", # pylint | ||
"PT", # flake8-pytest-style | ||
"PTH", # flake8-pathlib | ||
"PYI", # flake8-pyi | ||
"RET", # flake8-return | ||
"RSE", # flake8-raise | ||
"RUF005", # Consider iterable unpacking instead of concatenation | ||
"RUF006", # Store a reference to the return value of asyncio.create_task | ||
"RUF010", # Use explicit conversion flag | ||
"RUF013", # PEP 484 prohibits implicit Optional | ||
"RUF017", # Avoid quadratic list summation | ||
"RUF018", # Avoid assignment expressions in assert statements | ||
"RUF019", # Unnecessary key check before dictionary access | ||
# "RUF100", # Unused `noqa` directive; temporarily every now and then to clean them up | ||
"S102", # Use of exec detected | ||
"S103", # bad-file-permissions | ||
"S108", # hardcoded-temp-file | ||
"S306", # suspicious-mktemp-usage | ||
"S307", # suspicious-eval-usage | ||
"S313", # suspicious-xmlc-element-tree-usage | ||
"S314", # suspicious-xml-element-tree-usage | ||
"S315", # suspicious-xml-expat-reader-usage | ||
"S316", # suspicious-xml-expat-builder-usage | ||
"S317", # suspicious-xml-sax-usage | ||
"S318", # suspicious-xml-mini-dom-usage | ||
"S319", # suspicious-xml-pull-dom-usage | ||
"S320", # suspicious-xmle-tree-usage | ||
"S601", # paramiko-call | ||
"S602", # subprocess-popen-with-shell-equals-true | ||
"S604", # call-with-shell-equals-true | ||
"S608", # hardcoded-sql-expression | ||
"S609", # unix-command-wildcard-injection | ||
"SIM", # flake8-simplify | ||
"SLF", # flake8-self | ||
"SLOT", # flake8-slots | ||
"T100", # Trace found: {name} used | ||
"T20", # flake8-print | ||
"TC", # flake8-type-checking | ||
"TID", # Tidy imports | ||
"TRY", # tryceratops | ||
"UP", # pyupgrade | ||
"UP031", # Use format specifiers instead of percent format | ||
"UP032", # Use f-string instead of `format` call | ||
"W", # pycodestyle | ||
] | ||
|
||
ignore = [ | ||
"C901", # Temporarily | ||
"D202", # No blank lines allowed after function docstring | ||
"D203", # 1 blank line required before class docstring | ||
"D213", # Multi-line docstring summary should start at the second line | ||
"D406", # Section name should end with a newline | ||
"D407", # Section name underlining | ||
"E501", # line too long | ||
|
||
"PLC1901", # {existing} can be simplified to {replacement} as an empty string is falsey; too many false positives | ||
"PLR0904", # Too many public methods | ||
"PLR0911", # Too many return statements ({returns} > {max_returns}) | ||
"PLR0912", # Too many branches ({branches} > {max_branches}) | ||
"PLR0913", # Too many arguments to function call ({c_args} > {max_args}) | ||
"PLR0914", # Too many local variables | ||
"PLR0915", # Too many statements ({statements} > {max_statements}) | ||
"PLR0916", # Too many Boolean expressions | ||
"PLR0917", # Too many positional arguments | ||
"PLR1702", # Too many nested blocks | ||
"PLR2004", # Magic value used in comparison, consider replacing {value} with a constant variable | ||
"PLW2901", # Outer {outer_kind} variable {name} overwritten by inner {inner_kind} target | ||
"PT011", # pytest.raises({exception}) is too broad, set the `match` parameter or use a more specific exception | ||
"PT018", # Assertion should be broken down into multiple parts | ||
"RUF001", # String contains ambiguous unicode character. | ||
"RUF002", # Docstring contains ambiguous unicode character. | ||
"RUF003", # Comment contains ambiguous unicode character. | ||
"RUF015", # Prefer next(...) over single element slice | ||
"SIM102", # Use a single if statement instead of nested if statements | ||
"SIM103", # Return the condition {condition} directly | ||
"SIM108", # Use ternary operator {contents} instead of if-else-block | ||
"SIM115", # Use context handler for opening files | ||
|
||
# Moving imports into type-checking blocks can mess with pytest.patch() | ||
"TC001", # Move application import {} into a type-checking block | ||
"TC002", # Move third-party import {} into a type-checking block | ||
"TC003", # Move standard library import {} into a type-checking block | ||
|
||
"TRY003", # Avoid specifying long messages outside the exception class | ||
"TRY400", # Use `logging.exception` instead of `logging.error` | ||
|
||
# May conflict with the formatter, https://docs.astral.sh/ruff/formatter/#conflicting-lint-rules | ||
"W191", | ||
"E111", | ||
"E114", | ||
"E117", | ||
"D206", | ||
"D300", | ||
"Q", | ||
"COM812", | ||
"COM819", | ||
"ISC001", | ||
|
||
# Disabled because ruff does not understand type of __all__ generated by a function | ||
"PLE0605" | ||
] | ||
fixable = ["ALL"] | ||
|
||
[tool.ruff.lint.flake8-import-conventions.extend-aliases] | ||
voluptuous = "vol" | ||
"homeassistant.components.air_quality.PLATFORM_SCHEMA" = "AIR_QUALITY_PLATFORM_SCHEMA" | ||
"homeassistant.components.alarm_control_panel.PLATFORM_SCHEMA" = "ALARM_CONTROL_PANEL_PLATFORM_SCHEMA" | ||
"homeassistant.components.binary_sensor.PLATFORM_SCHEMA" = "BINARY_SENSOR_PLATFORM_SCHEMA" | ||
"homeassistant.components.button.PLATFORM_SCHEMA" = "BUTTON_PLATFORM_SCHEMA" | ||
"homeassistant.components.calendar.PLATFORM_SCHEMA" = "CALENDAR_PLATFORM_SCHEMA" | ||
"homeassistant.components.camera.PLATFORM_SCHEMA" = "CAMERA_PLATFORM_SCHEMA" | ||
"homeassistant.components.climate.PLATFORM_SCHEMA" = "CLIMATE_PLATFORM_SCHEMA" | ||
"homeassistant.components.conversation.PLATFORM_SCHEMA" = "CONVERSATION_PLATFORM_SCHEMA" | ||
"homeassistant.components.cover.PLATFORM_SCHEMA" = "COVER_PLATFORM_SCHEMA" | ||
"homeassistant.components.date.PLATFORM_SCHEMA" = "DATE_PLATFORM_SCHEMA" | ||
"homeassistant.components.datetime.PLATFORM_SCHEMA" = "DATETIME_PLATFORM_SCHEMA" | ||
"homeassistant.components.device_tracker.PLATFORM_SCHEMA" = "DEVICE_TRACKER_PLATFORM_SCHEMA" | ||
"homeassistant.components.event.PLATFORM_SCHEMA" = "EVENT_PLATFORM_SCHEMA" | ||
"homeassistant.components.fan.PLATFORM_SCHEMA" = "FAN_PLATFORM_SCHEMA" | ||
"homeassistant.components.geo_location.PLATFORM_SCHEMA" = "GEO_LOCATION_PLATFORM_SCHEMA" | ||
"homeassistant.components.humidifier.PLATFORM_SCHEMA" = "HUMIDIFIER_PLATFORM_SCHEMA" | ||
"homeassistant.components.image.PLATFORM_SCHEMA" = "IMAGE_PLATFORM_SCHEMA" | ||
"homeassistant.components.image_processing.PLATFORM_SCHEMA" = "IMAGE_PROCESSING_PLATFORM_SCHEMA" | ||
"homeassistant.components.lawn_mower.PLATFORM_SCHEMA" = "LAWN_MOWER_PLATFORM_SCHEMA" | ||
"homeassistant.components.light.PLATFORM_SCHEMA" = "LIGHT_PLATFORM_SCHEMA" | ||
"homeassistant.components.lock.PLATFORM_SCHEMA" = "LOCK_PLATFORM_SCHEMA" | ||
"homeassistant.components.media_player.PLATFORM_SCHEMA" = "MEDIA_PLAYER_PLATFORM_SCHEMA" | ||
"homeassistant.components.notify.PLATFORM_SCHEMA" = "NOTIFY_PLATFORM_SCHEMA" | ||
"homeassistant.components.number.PLATFORM_SCHEMA" = "NUMBER_PLATFORM_SCHEMA" | ||
"homeassistant.components.remote.PLATFORM_SCHEMA" = "REMOTE_PLATFORM_SCHEMA" | ||
"homeassistant.components.scene.PLATFORM_SCHEMA" = "SCENE_PLATFORM_SCHEMA" | ||
"homeassistant.components.select.PLATFORM_SCHEMA" = "SELECT_PLATFORM_SCHEMA" | ||
"homeassistant.components.sensor.PLATFORM_SCHEMA" = "SENSOR_PLATFORM_SCHEMA" | ||
"homeassistant.components.siren.PLATFORM_SCHEMA" = "SIREN_PLATFORM_SCHEMA" | ||
"homeassistant.components.stt.PLATFORM_SCHEMA" = "STT_PLATFORM_SCHEMA" | ||
"homeassistant.components.switch.PLATFORM_SCHEMA" = "SWITCH_PLATFORM_SCHEMA" | ||
"homeassistant.components.text.PLATFORM_SCHEMA" = "TEXT_PLATFORM_SCHEMA" | ||
"homeassistant.components.time.PLATFORM_SCHEMA" = "TIME_PLATFORM_SCHEMA" | ||
"homeassistant.components.todo.PLATFORM_SCHEMA" = "TODO_PLATFORM_SCHEMA" | ||
"homeassistant.components.tts.PLATFORM_SCHEMA" = "TTS_PLATFORM_SCHEMA" | ||
"homeassistant.components.vacuum.PLATFORM_SCHEMA" = "VACUUM_PLATFORM_SCHEMA" | ||
"homeassistant.components.valve.PLATFORM_SCHEMA" = "VALVE_PLATFORM_SCHEMA" | ||
"homeassistant.components.update.PLATFORM_SCHEMA" = "UPDATE_PLATFORM_SCHEMA" | ||
"homeassistant.components.wake_word.PLATFORM_SCHEMA" = "WAKE_WORD_PLATFORM_SCHEMA" | ||
"homeassistant.components.water_heater.PLATFORM_SCHEMA" = "WATER_HEATER_PLATFORM_SCHEMA" | ||
"homeassistant.components.weather.PLATFORM_SCHEMA" = "WEATHER_PLATFORM_SCHEMA" | ||
"homeassistant.core.DOMAIN" = "HOMEASSISTANT_DOMAIN" | ||
"homeassistant.helpers.area_registry" = "ar" | ||
"homeassistant.helpers.category_registry" = "cr" | ||
"homeassistant.helpers.config_validation" = "cv" | ||
"homeassistant.helpers.device_registry" = "dr" | ||
"homeassistant.helpers.entity_registry" = "er" | ||
"homeassistant.helpers.floor_registry" = "fr" | ||
"homeassistant.helpers.issue_registry" = "ir" | ||
"homeassistant.helpers.label_registry" = "lr" | ||
"homeassistant.util.dt" = "dt_util" | ||
|
||
[tool.ruff.lint.flake8-pytest-style] | ||
fixture-parentheses = false | ||
mark-parentheses = false | ||
|
||
[tool.ruff.lint.flake8-tidy-imports.banned-api] | ||
"async_timeout".msg = "use asyncio.timeout instead" | ||
"pytz".msg = "use zoneinfo instead" | ||
|
||
[tool.ruff.lint.isort] | ||
force-sort-within-sections = true | ||
known-first-party = [ | ||
"homeassistant", | ||
] | ||
combine-as-imports = true | ||
split-on-trailing-comma = false | ||
|
||
[tool.ruff.lint.mccabe] | ||
max-complexity = 25 | ||
|
||
[tool.ruff.lint.pydocstyle] | ||
property-decorators = ["propcache.cached_property"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
aiohttp_cors | ||
mypy | ||
ruff | ||
types-python-dateutil | ||
types-PyMySQL | ||
types-PyYAML | ||
types-pyRFC3339 | ||
types-requests |