Skip to content

Commit c8cbc93

Browse files
committedJul 7, 2023
Migrate to use repo-config
Many files were updated from the generated files by repo-config. There are some changes that need some clarification: GitHub workflow: - The explicit cache is not needed because the Python action supports caching natively. - The announcement discussion for new release is not created anymore as it was mainly noise, nobody ever used it. pyproject.toml: - We ping build dependencies, as they don't affect users. - pytest-cov dependency is removed because we are not actually producing coverage reports. Signed-off-by: Leandro Lucarella <[email protected]>
1 parent 9671a6e commit c8cbc93

16 files changed

+153
-452
lines changed
 

‎.cookiecutter-replay.json

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
{
2+
"cookiecutter": {
3+
"type": "lib",
4+
"name": "sdk",
5+
"description": "A development kit to interact with the Frequenz development platform",
6+
"title": "Frequenz Python SDK",
7+
"keywords": "microgrid,actor,ml,machine-learning,ai",
8+
"github_org": "frequenz-floss",
9+
"license": "MIT",
10+
"author_name": "Frequenz Energy-as-a-Service GmbH",
11+
"author_email": "floss@frequenz.com",
12+
"python_package": "frequenz.sdk",
13+
"pypi_package_name": "frequenz-sdk",
14+
"github_repo_name": "frequenz-sdk-python",
15+
"default_codeowners": "@frequenz-floss/python-sdk-team",
16+
"_extensions": [
17+
"jinja2_time.TimeExtension",
18+
"local_extensions.default_codeowners",
19+
"local_extensions.github_repo_name",
20+
"local_extensions.keywords",
21+
"local_extensions.pypi_package_name",
22+
"local_extensions.python_package",
23+
"local_extensions.src_path",
24+
"local_extensions.title"
25+
],
26+
"_template": "gh:frequenz-floss/frequenz-repo-config-python"
27+
}
28+
}

‎.github/RELEASE_NOTES.template.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
1-
# Release Notes
1+
# Frequenz Python SDK Release Notes
22

33
## Summary
44

55
<!-- Here goes a general summary of what this release is about -->
66

77
## Upgrading
88

9-
<!-- Here goes notes on how to upgrade from previous versions, including deprecations and what they should be replaced with -->
9+
<!-- Here goes notes on how to upgrade from previous versions, including deprecations and what they should be replaced with -->
1010

1111
## New Features
1212

‎.github/labeler.yml

+2-5
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020
- "src/frequenz/sdk/_data_ingestion/**"
2121
- "src/frequenz/sdk/timeseries/**"
2222

23-
"part:docs":
23+
"part:docs":
2424
- "**/*.md"
2525
- "docs/**"
2626
- "examples/**"
@@ -40,12 +40,9 @@
4040
- "**/*.toml"
4141
- "**/*.yaml"
4242
- "**/*.yml"
43-
- "*requirements*.txt"
4443
- ".git*"
4544
- ".git*/**"
45+
- "docs/*.py"
4646
- CODEOWNERS
4747
- MANIFEST.in
48-
- docs/mkdocstrings_autoapi.py
4948
- noxfile.py
50-
- setup.cfg
51-
- setup.py

‎.github/workflows/ci.yaml

+32-26
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
1-
name: frequenz-sdk-python
1+
name: CI
22

33
on:
44
merge_group:
55
pull_request:
66
push:
7+
# We need to explicitly include tags because otherwise when adding
8+
# `branches-ignore` it will only trigger on branches.
79
tags:
810
- '*'
911
branches-ignore:
@@ -14,10 +16,15 @@ on:
1416
workflow_dispatch:
1517

1618
env:
19+
# Please make sure this version is included in the `matrix`, as the
20+
# `matrix` section can't use `env`, so it must be entered manually
1721
DEFAULT_PYTHON_VERSION: '3.11'
22+
# It would be nice to be able to also define a DEFAULT_UBUNTU_VERSION
23+
# but sadly `env` can't be used either in `runs-on`.
1824

1925
jobs:
20-
test:
26+
nox:
27+
name: Test with nox
2128
strategy:
2229
fail-fast: false
2330
matrix:
@@ -37,23 +44,19 @@ jobs:
3744
python-version: ${{ matrix.python }}
3845
cache: 'pip'
3946

40-
- uses: actions/cache@v3
41-
with:
42-
path: ~/.cache/pip
43-
key: ${{ runner.os }}-${{ matrix.python-version }}-pip-${{ hashFiles('minimum-requirements-ci.txt', 'pyproject.toml') }}
44-
restore-keys: |
45-
${{ runner.os }}-${{ matrix.python-version }}-pip-
46-
4747
- name: Install required Python packages
4848
run: |
4949
python -m pip install --upgrade pip
50-
python -m pip install nox toml
50+
python -m pip install -e .[dev-noxfile]
5151
52-
- name: run nox
52+
- name: Run nox
53+
# To speed things up a bit we use the speciall ci_checks_max session
54+
# that uses the same venv to run multiple linting sessions
5355
run: nox -e ci_checks_max pytest_min
5456
timeout-minutes: 10
5557

56-
build-dist:
58+
build:
59+
name: Build distribution packages
5760
runs-on: ubuntu-20.04
5861
steps:
5962
- name: Fetch sources
@@ -62,25 +65,26 @@ jobs:
6265
- name: Set up Python
6366
uses: actions/setup-python@v4
6467
with:
65-
python-version: '3.11'
68+
python-version: ${{ env.DEFAULT_PYTHON_VERSION }}
6669
cache: 'pip'
6770

68-
- name: Install build dependencies
71+
- name: Install required Python packages
6972
run: |
7073
python -m pip install -U pip
7174
python -m pip install -U build
7275
7376
- name: Build the source and binary distribution
7477
run: python -m build
7578

76-
- name: Upload dist files
79+
- name: Upload distribution files
7780
uses: actions/upload-artifact@v3
7881
with:
79-
name: frequenz-sdk-python-dist
82+
name: dist-packages
8083
path: dist/
8184
if-no-files-found: error
8285

83-
test-generate-docs:
86+
test-docs:
87+
name: Test documentation website generation
8488
if: github.event_name != 'push'
8589
runs-on: ubuntu-20.04
8690
steps:
@@ -99,7 +103,7 @@ jobs:
99103
- name: Install build dependencies
100104
run: |
101105
python -m pip install -U pip
102-
python -m pip install .[docs-gen]
106+
python -m pip install .[dev-mkdocs]
103107
104108
- name: Generate the documentation
105109
env:
@@ -111,12 +115,13 @@ jobs:
111115
- name: Upload site
112116
uses: actions/upload-artifact@v3
113117
with:
114-
name: frequenz-channels-python-site
118+
name: docs-site
115119
path: site/
116120
if-no-files-found: error
117121

118122
publish-docs:
119-
needs: ["test", "build-dist"]
123+
name: Publish documentation website to GitHub pages
124+
needs: ["nox", "build"]
120125
if: github.event_name == 'push'
121126
runs-on: ubuntu-20.04
122127
permissions:
@@ -178,7 +183,7 @@ jobs:
178183
if: steps.mike-metadata.outputs.version
179184
run: |
180185
python -m pip install -U pip
181-
python -m pip install .[docs-gen]
186+
python -m pip install .[dev-mkdocs]
182187
183188
- name: Fetch the gh-pages branch
184189
if: steps.mike-metadata.outputs.version
@@ -193,6 +198,7 @@ jobs:
193198
mike deploy --push --update-aliases "$VERSION" $ALIASES
194199
195200
create-github-release:
201+
name: Create GitHub release
196202
needs: ["publish-docs"]
197203
# Create a release only on tags creation
198204
if: github.event_name == 'push' && startsWith(github.ref, 'refs/tags/v')
@@ -203,10 +209,10 @@ jobs:
203209
discussions: write
204210
runs-on: ubuntu-20.04
205211
steps:
206-
- name: Download dist files
212+
- name: Download distribution files
207213
uses: actions/download-artifact@v3
208214
with:
209-
name: frequenz-sdk-python-dist
215+
name: dist-packages
210216
path: dist
211217

212218
- name: Download RELEASE_NOTES.md
@@ -230,7 +236,6 @@ jobs:
230236
if echo "$REF_NAME" | grep -- -; then extra_opts=" --prerelease"; fi
231237
gh release create \
232238
-R "$REPOSITORY" \
233-
--discussion-category announcements \
234239
--notes-file RELEASE_NOTES.md \
235240
--generate-notes \
236241
$extra_opts \
@@ -242,17 +247,18 @@ jobs:
242247
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
243248

244249
publish-to-pypi:
250+
name: Publish packages to PyPI
245251
needs: ["create-github-release"]
246252
runs-on: ubuntu-20.04
247253
permissions:
248254
# For trusted publishing. See:
249255
# https://blog.pypi.org/posts/2023-04-20-introducing-trusted-publishers/
250256
id-token: write
251257
steps:
252-
- name: Download dist files
258+
- name: Download distribution files
253259
uses: actions/download-artifact@v3
254260
with:
255-
name: frequenz-sdk-python-dist
261+
name: dist-packages
256262
path: dist
257263

258264
- name: Publish the Python distribution to PyPI

‎.github/workflows/labeler.yml

+8-9
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,5 @@
11
name: Pull Request Labeler
22

3-
# XXX: !!! SECURITY WARNING !!!
4-
# pull_request_target has write access to the repo, and can read secrets. We
5-
# need to audit any external actions executed in this workflow and make sure no
6-
# checked out code is run (not even installing dependencies, as installing
7-
# dependencies usually can execute pre/post-install scripts). We should also
8-
# only use hashes to pick the action to execute (instead of tags or branches).
9-
# For more details read:
10-
# https://securitylab.github.com/research/github-actions-preventing-pwn-requests/
113
on: [pull_request_target]
124

135
jobs:
@@ -18,7 +10,14 @@ jobs:
1810
runs-on: ubuntu-latest
1911
steps:
2012
- name: Labeler
21-
# Only use hashes, see the security comment above
13+
# XXX: !!! SECURITY WARNING !!!
14+
# pull_request_target has write access to the repo, and can read secrets. We
15+
# need to audit any external actions executed in this workflow and make sure no
16+
# checked out code is run (not even installing dependencies, as installing
17+
# dependencies usually can execute pre/post-install scripts). We should also
18+
# only use hashes to pick the action to execute (instead of tags or branches).
19+
# For more details read:
20+
# https://securitylab.github.com/research/github-actions-preventing-pwn-requests/
2221
uses: actions/labeler@0967ca812e7fdc8f5f71402a1b486d5bd061fe20 # 4.2.0
2322
with:
2423
repo-token: "${{ secrets.GITHUB_TOKEN }}"

‎.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,9 @@ venv/
114114
ENV/
115115
env.bak/
116116
venv.bak/
117+
# direnv https://github.com/direnv/direnv
118+
.envrc
119+
.direnv/
117120

118121
# Spyder project settings
119122
.spyderproject

‎CONTRIBUTING.md

+15-16
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
# Contributing to `frequenz-sdk`
1+
# Contributing to Frequenz Python SDK
22

33
## Build
44

@@ -28,7 +28,7 @@ If you don't want to install all the dependencies, you can also use `nox` to
2828
run the tests and other checks creating its own virtual environments:
2929

3030
```sh
31-
python -m pip install nox toml
31+
python -m pip install .[dev-noxfile]
3232
nox
3333
```
3434

@@ -41,23 +41,23 @@ For a better development test cycle you can install the runtime and test
4141
dependencies and run `pytest` manually.
4242

4343
```sh
44-
python -m pip install .[pytest] # included in .[dev] too
44+
python -m pip install .[dev-pytest] # included in .[dev] too
4545

4646
# And for example
47-
pytest tests/test_sdk.py
47+
pytest tests/test_*.py
4848
```
4949

5050
Or you can use `nox`:
5151

5252
```sh
53-
nox -R -s pytest -- test/test_sdk.py
53+
nox -R -s pytest -- test/test_*.py
5454
```
5555

5656
The same appliest to `pylint` or `mypy` for example:
5757

5858
```sh
59-
nox -R -s pylint -- test/test_sdk.py
60-
nox -R -s mypy -- test/test_sdk.py
59+
nox -R -s pylint -- test/test_*.py
60+
nox -R -s mypy -- test/test_*.py
6161
```
6262

6363
### Building the documentation
@@ -66,7 +66,7 @@ To build the documentation, first install the dependencies (if you didn't
6666
install all `dev` dependencies):
6767

6868
```sh
69-
python -m pip install -e .[docs-gen]
69+
python -m pip install -e .[dev-mkdocs]
7070
```
7171

7272
Then you can build the documentation (it will be written in the `site/`
@@ -122,28 +122,27 @@ These are the steps to create a new release:
122122
1. Get the latest head you want to create a release from.
123123

124124
2. Update the `RELEASE_NOTES.md` file if it is not complete, up to date, and
125-
clean from template comments (`<!-- ... ->`) and empty sections. Submit
126-
a pull request if an update is needed, wait until it is merged, and update
127-
the latest head you want to create a release from to get the new merged pull
125+
remove template comments (`<!-- ... ->`) and empty sections. Submit a pull
126+
request if an update is needed, wait until it is merged, and update the
127+
latest head you want to create a release from to get the new merged pull
128128
request.
129129

130130
3. Create a new signed tag using the release notes and
131131
a [semver](https://semver.org/) compatible version number with a `v` prefix,
132132
for example:
133133

134134
```sh
135-
git tag -s -F RELEASE_NOTES.md v0.0.1
135+
git tag -s --cleanup=whitespace -F RELEASE_NOTES.md v0.0.1
136136
```
137137

138138
4. Push the new tag.
139139

140140
5. A GitHub action will test the tag and if all goes well it will create
141141
a [GitHub
142142
Release](https://github.com/frequenz-floss/frequenz-sdk-python/releases),
143-
create a new
144-
[announcement](https://github.com/frequenz-floss/frequenz-sdk-python/discussions/categories/announcements)
145-
about the release, and upload a new package to
146-
[PyPI](https://pypi.org/project/frequenz-sdk/) automatically.
143+
and upload a new package to
144+
[PyPI](https://pypi.org/project/frequenz-sdk/)
145+
automatically.
147146

148147
6. Once this is done, reset the `RELEASE_NOTES.md` with the template:
149148

‎LICENSE

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
MIT License
22

3-
Copyright (c) 2022 Frequenz Energy-as-a-Service GmbH
3+
Copyright © 2022 Frequenz Energy-as-a-Service GmbH
44

55
Permission is hereby granted, free of charge, to any person obtaining a copy
66
of this software and associated documentation files (the "Software"), to deal

‎README.md

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@
44
[![PyPI Package](https://img.shields.io/pypi/v/frequenz-sdk)](https://pypi.org/project/frequenz-sdk/)
55
[![Docs](https://img.shields.io/badge/docs-latest-informational)](https://frequenz-floss.github.io/frequenz-sdk-python/)
66

7+
## Introduction
8+
79
A development kit to interact with the Frequenz development platform.
810

911
## Quick Start

‎RELEASE_NOTES.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
# Release Notes
1+
# Frequenz Python SDK Release Notes
22

33
## Summary
44

‎docs/SUMMARY.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
* [Home](index.md)
22
* [API Reference](reference/)
3-
* [Development](CONTRIBUTING.md)
3+
* [Contributing](CONTRIBUTING.md)

‎docs/mkdocstrings_autoapi.py

+3-55
Original file line numberDiff line numberDiff line change
@@ -1,60 +1,8 @@
11
# License: MIT
22
# Copyright © 2022 Frequenz Energy-as-a-Service GmbH
33

4-
"""Generate the code reference pages.
4+
"""Generate the code reference pages."""
55

6-
Based on the recipe at:
7-
https://mkdocstrings.github.io/recipes/#automatic-code-reference-pages
8-
"""
6+
from frequenz.repo.config import mkdocs
97

10-
from pathlib import Path
11-
from typing import Tuple
12-
13-
import mkdocs_gen_files
14-
15-
SRC_PATH = "src"
16-
DST_PATH = "reference"
17-
18-
19-
def is_internal(path_parts: Tuple[str, ...]) -> bool:
20-
"""Tell if the path is internal judging by the parts.
21-
22-
Args:
23-
path_parts: Path.parts of the path to check.
24-
25-
Returns:
26-
True if the path is internal.
27-
"""
28-
29-
def with_underscore_not_init(part: str) -> bool:
30-
return part.startswith("_") and part != "__init__"
31-
32-
return any(p for p in path_parts if with_underscore_not_init(p))
33-
34-
35-
# type ignore because mkdocs_gen_files uses a very weird module-level
36-
# __getattr__() which messes up the type system
37-
nav = mkdocs_gen_files.Nav() # type: ignore
38-
39-
for path in sorted(Path(SRC_PATH).rglob("*.py")):
40-
module_path = path.relative_to(SRC_PATH).with_suffix("")
41-
42-
doc_path = path.relative_to(SRC_PATH).with_suffix(".md")
43-
full_doc_path = Path(DST_PATH, doc_path)
44-
parts = tuple(module_path.parts)
45-
if is_internal(parts):
46-
continue
47-
if parts[-1] == "__init__":
48-
doc_path = doc_path.with_name("index.md")
49-
full_doc_path = full_doc_path.with_name("index.md")
50-
parts = parts[:-1]
51-
52-
nav[parts] = doc_path.as_posix()
53-
54-
with mkdocs_gen_files.open(full_doc_path, "w") as output_file:
55-
output_file.write(f"::: {'.'.join(parts)}\n")
56-
57-
mkdocs_gen_files.set_edit_path(full_doc_path, Path("..") / path)
58-
59-
with mkdocs_gen_files.open(Path(DST_PATH) / "SUMMARY.md", "w") as nav_file:
60-
nav_file.writelines(nav.build_literate_nav())
8+
mkdocs.generate_python_api_pages("src", "reference")

‎docs/overrides/main.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
{% block outdated %}
44
You're not viewing the latest (stable) version.
5-
<a href="{{ '../' ~ base_url }}">
5+
<a href="{{ '../' ~ base_url }}">
66
<strong>Click here to go to latest (stable) version</strong>
77
</a>
88
{% endblock %}

‎mkdocs.yml

+7-6
Original file line numberDiff line numberDiff line change
@@ -2,13 +2,14 @@
22
# For details see: https://www.mkdocs.org/user-guide/configuration/
33

44
# Project information
5-
site_name: Frequenz Python SDK
6-
site_description: Frequenz Python SDK.
7-
site_author: Frequenz Energy-as-a-Service GmbH
8-
copyright: Frequenz Energy-as-a-Service GmbH
5+
site_name: "Frequenz Python SDK"
6+
site_description: "A development kit to interact with the Frequenz development platform"
7+
site_author: "Frequenz Energy-as-a-Service GmbH"
8+
copyright: "Copyright © 2022 Frequenz Energy-as-a-Service GmbH"
99
repo_name: "frequenz-sdk-python"
1010
repo_url: "https://github.com/frequenz-floss/frequenz-sdk-python"
1111
edit_uri: "edit/v0.x.x/docs/"
12+
strict: true # Treat warnings as errors
1213

1314
# Build directories
1415
theme:
@@ -88,7 +89,7 @@ plugins:
8889
handlers:
8990
python:
9091
options:
91-
paths: [src]
92+
paths: ["src"]
9293
docstring_section_style: spacy
9394
merge_init_into_class: false
9495
show_category_heading: true
@@ -102,6 +103,6 @@ plugins:
102103

103104
# Preview controls
104105
watch:
105-
- src
106+
- "src"
106107
- README.md
107108
- CONTRIBUTING.md

‎noxfile.py

+3-287
Original file line numberDiff line numberDiff line change
@@ -1,292 +1,8 @@
11
# License: MIT
22
# Copyright © 2022 Frequenz Energy-as-a-Service GmbH
33

4-
"""Automation for code quality checks and unit tests for the Frequenz SDK.
4+
"""Configuration file for nox."""
55

6-
This file specified all the checks that can be run from command line invocations
7-
of `nox`.
6+
from frequenz.repo.config import RepositoryType, nox
87

9-
The following checks are performed:
10-
11-
1. `formatting` :: checks that the code is formatted with `black` and the imports
12-
are sorted with `isort`.
13-
2. `mypy` :: type checks all source files with `mypy --strict`.
14-
3. `pylint` :: lints all source files with `pylint`.
15-
4. `docstrings` :: checks that all public functions have docstrings with
16-
1. a one-line imperative description at the top, followed by additional
17-
description, using `pydocstyle`.
18-
2. function parameters, return values, and raised exceptions are documented
19-
following the google style guide for these items:
20-
https://google.github.io/styleguide/pyguide.html#doc-function-args, using
21-
`darglint`.
22-
5. `pytest_min` :: run all unittests using `pytest`, with the oldest supported
23-
versions of all dependencies installed.
24-
6. `pytest_max` :: run all unittests using `pytest`, with the latest supported
25-
versions of all dependencies installed.
26-
27-
Usage:
28-
29-
1. Run all checks in a *new* venv.
30-
31-
nox
32-
33-
2. Run all checks in an *exising* venv. This would be much faster if venv for
34-
all tests exist already. If they don't exist, new venvs will be created.
35-
36-
nox -R
37-
38-
3. Run a subset of available checks:
39-
40-
nox -e mypy pylint # create new venvs for specified checks.
41-
nox -R -e mypy pylint # reuse venvs for specified checks if available.
42-
43-
4. The `pytest_min` and `pytest_max` checks run `pytest` on all available tests,
44-
including test coverage generation. But this can be slow for fast local
45-
test-devlop cycles, and so `pytest` can also be invoked with optional custom
46-
arguments, in which case, only the specified arguments are passed to
47-
`pytest`. This can be done as follows:
48-
49-
nox -R -e pytest_min [-- <args for pytest>]
50-
nox -R -e pytest_min -- -s -x tests/timeseries/test_logical_meter.py
51-
"""
52-
53-
from __future__ import annotations
54-
55-
from typing import Any, Iterable
56-
57-
import nox
58-
import toml
59-
60-
DEFAULT_PATH_PACKAGES = {
61-
"benchmarks": "benchmarks",
62-
"docs": "docs",
63-
"examples": "examples",
64-
"src": "frequenz.sdk",
65-
"tests": "tests",
66-
"noxfile.py": "noxfile",
67-
}
68-
"""A list of path to be used by default and its corresponding package name.
69-
70-
The package name is needed for mypy, as it takes packages when full import
71-
checking needs to be done.
72-
"""
73-
74-
75-
def min_dependencies() -> list[str]:
76-
"""Extract the minimum dependencies from pyproject.toml.
77-
78-
Raises:
79-
RuntimeError: If minimun dependencies are not properly
80-
set in pyproject.toml.
81-
82-
Returns:
83-
the minimun dependencies defined in pyproject.toml.
84-
85-
"""
86-
with open("pyproject.toml", "r", encoding="utf-8") as toml_file:
87-
data = toml.load(toml_file)
88-
89-
dependencies = data.get("project", {}).get("dependencies", {})
90-
if not dependencies:
91-
raise RuntimeError(f"No dependencies found in file: {toml_file.name}")
92-
93-
min_deps: list[str] = []
94-
for dep in dependencies:
95-
min_dep = dep.split(",")[0]
96-
if any(op in min_dep for op in (">=", "==")):
97-
min_deps.append(min_dep.replace(">=", "=="))
98-
else:
99-
raise RuntimeError(f"Minimum requirement is not set: {dep}")
100-
return min_deps
101-
102-
103-
def _source_file_paths(session: nox.Session) -> list[str]:
104-
"""Return the file paths to run the checks on.
105-
106-
If positional arguments are present in the nox session, we use those as the
107-
file paths, and if not, we use all source files.
108-
109-
Args:
110-
session: the nox session.
111-
112-
Returns:
113-
the file paths to run the checks on.
114-
"""
115-
if session.posargs:
116-
return session.posargs
117-
return list(DEFAULT_PATH_PACKAGES.keys())
118-
119-
120-
# Run all checks except `ci_checks` by default. When running locally with just
121-
# `nox` or `nox -R`, these are the checks that will run.
122-
nox.options.sessions = [
123-
"formatting",
124-
"mypy",
125-
"pylint",
126-
"docstrings",
127-
"pytest_min",
128-
"pytest_max",
129-
]
130-
131-
132-
@nox.session
133-
def ci_checks_max(session: nox.Session) -> None:
134-
"""Run all checks with max dependencies in a single session.
135-
136-
This is faster than running the checks separately, so it is suitable for CI.
137-
138-
This does NOT run pytest_min, so that needs to be run separately as well.
139-
140-
Args:
141-
session: the nox session.
142-
"""
143-
session.install("-e", ".[dev]")
144-
145-
formatting(session, False)
146-
mypy(session, False)
147-
pylint(session, False)
148-
docstrings(session, False)
149-
pytest_max(session, False)
150-
151-
152-
@nox.session
153-
def formatting(session: nox.Session, install_deps: bool = True) -> None:
154-
"""Check code formatting with black and isort.
155-
156-
Args:
157-
session: the nox session.
158-
install_deps: True if dependencies should be installed.
159-
"""
160-
if install_deps:
161-
session.install("-e", ".[format]")
162-
163-
paths = _source_file_paths(session)
164-
session.run("black", "--check", *paths)
165-
session.run("isort", "--diff", "--check", *paths)
166-
167-
168-
@nox.session
169-
def mypy(session: nox.Session, install_deps: bool = True) -> None:
170-
"""Check type hints with mypy.
171-
172-
Args:
173-
session: the nox session.
174-
install_deps: True if dependencies should be installed.
175-
"""
176-
if install_deps:
177-
# install the package itself as editable, so that it is possible to do
178-
# fast local tests with `nox -R -e mypy`.
179-
session.install("-e", ".[mypy]")
180-
181-
def _flatten(iterable: Iterable[Iterable[Any]]) -> Iterable[Any]:
182-
return [item for sublist in iterable for item in sublist]
183-
184-
args = (
185-
session.posargs
186-
if session.posargs
187-
else _flatten(("-p", p) for p in DEFAULT_PATH_PACKAGES.values())
188-
)
189-
190-
session.run(
191-
"mypy",
192-
"--install-types",
193-
"--namespace-packages",
194-
"--non-interactive",
195-
"--explicit-package-bases",
196-
"--strict",
197-
*args,
198-
)
199-
200-
201-
@nox.session
202-
def pylint(session: nox.Session, install_deps: bool = True) -> None:
203-
"""Check for code smells with pylint.
204-
205-
Args:
206-
session: the nox session.
207-
install_deps: True if dependencies should be installed.
208-
"""
209-
if install_deps:
210-
# install the package itself as editable, so that it is possible to do
211-
# fast local tests with `nox -R -e pylint`.
212-
session.install("-e", ".[pylint]")
213-
214-
paths = _source_file_paths(session)
215-
session.run(
216-
"pylint",
217-
"--extension-pkg-whitelist=pydantic",
218-
*paths,
219-
)
220-
221-
222-
@nox.session
223-
def docstrings(session: nox.Session, install_deps: bool = True) -> None:
224-
"""Check docstring tone with pydocstyle and param descriptions with darglint.
225-
226-
Args:
227-
session: the nox session.
228-
install_deps: True if dependencies should be installed.
229-
"""
230-
if install_deps:
231-
session.install("-e", ".[docs-lint]")
232-
233-
paths = _source_file_paths(session)
234-
session.run("pydocstyle", *paths)
235-
236-
# Darglint checks that function argument and return values are documented.
237-
# This is needed only for the `src` dir, so we exclude the other top level
238-
# dirs that contain code, unless some paths were specified by argument, in
239-
# which case we use those untouched.
240-
darglint_paths = session.posargs or filter(
241-
lambda path: not (path.startswith("tests") or path.startswith("benchmarks")),
242-
_source_file_paths(session),
243-
)
244-
session.run(
245-
"darglint",
246-
"-v2", # for verbose error messages.
247-
*darglint_paths,
248-
)
249-
250-
251-
@nox.session
252-
def pytest_max(session: nox.Session, install_deps: bool = True) -> None:
253-
"""Test the code against max dependency versions with pytest.
254-
255-
Args:
256-
session: the nox session.
257-
install_deps: True if dependencies should be installed.
258-
"""
259-
if install_deps:
260-
# install the package itself as editable, so that it is possible to do
261-
# fast local tests with `nox -R -e pytest_max`.
262-
session.install("-e", ".[pytest]")
263-
264-
_pytest_impl(session, "max")
265-
266-
267-
@nox.session
268-
def pytest_min(session: nox.Session, install_deps: bool = True) -> None:
269-
"""Test the code against min dependency versions with pytest.
270-
271-
Args:
272-
session: the nox session.
273-
install_deps: True if dependencies should be installed.
274-
"""
275-
if install_deps:
276-
# install the package itself as editable, so that it is possible to do
277-
# fast local tests with `nox -R -e pytest_min`.
278-
session.install("-e", ".[pytest]", *min_dependencies())
279-
280-
_pytest_impl(session, "min")
281-
282-
283-
def _pytest_impl(session: nox.Session, max_or_min_deps: str) -> None:
284-
session.run(
285-
"pytest",
286-
"-W=all",
287-
"-vv",
288-
"--cov=frequenz.sdk",
289-
"--cov-report=term",
290-
f"--cov-report=html:.htmlcov-{max_or_min_deps}",
291-
*session.posargs,
292-
)
8+
nox.configure(RepositoryType.LIB)

‎pyproject.toml

+44-42
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,17 @@
1+
# License: MIT
2+
# Copyright © 2022 Frequenz Energy-as-a-Service GmbH
3+
14
[build-system]
25
requires = [
3-
"setuptools >= 65.3.0, < 66",
4-
"setuptools_scm[toml] >= 7.0.5, < 8",
5-
"wheel",
6+
"setuptools == 67.7.2",
7+
"setuptools_scm[toml] == 7.1.0",
8+
"frequenz-repo-config[lib] == 0.3.0",
69
]
710
build-backend = "setuptools.build_meta"
811

912
[project]
1013
name = "frequenz-sdk"
11-
description = "Frequenz Python SDK"
14+
description = "A development kit to interact with the Frequenz development platform"
1215
readme = "README.md"
1316
license = { text = "MIT" }
1417
keywords = ["frequenz", "sdk", "microgrid", "actor"]
@@ -19,6 +22,7 @@ classifiers = [
1922
"Programming Language :: Python :: 3",
2023
"Programming Language :: Python :: 3 :: Only",
2124
"Topic :: Software Development :: Libraries",
25+
"Typing :: Typed",
2226
]
2327
requires-python = ">= 3.11, < 4"
2428
dependencies = [
@@ -43,64 +47,65 @@ name = "Frequenz Energy-as-a-Service GmbH"
4347
email = "floss@frequenz.com"
4448

4549
[project.optional-dependencies]
46-
docs-gen = [
50+
dev-docstrings = [
51+
"pydocstyle == 6.3.0",
52+
"darglint == 1.8.1",
53+
"tomli == 2.0.1", # Needed by pydocstyle to read pyproject.toml
54+
]
55+
dev-examples = ["polars == 0.18.6"]
56+
dev-formatting = ["black == 23.3.0", "isort == 5.12.0"]
57+
dev-mkdocs = [
4758
"mike == 1.1.2",
4859
"mkdocs-gen-files == 0.5.0",
4960
"mkdocs-literate-nav == 0.6.0",
5061
"mkdocs-material == 9.1.18",
5162
"mkdocs-section-index == 0.3.5",
5263
"mkdocstrings[python] == 0.22.0",
64+
"frequenz-repo-config[lib] == 0.3.0",
5365
]
54-
docs-lint = [
55-
"pydocstyle == 6.3.0",
56-
"darglint == 1.8.1",
57-
"tomli == 2.0.1", # Needed by pydocstyle to read pyproject.toml
66+
dev-mypy = [
67+
"mypy == 1.4.1",
68+
"grpc-stubs == 1.24.12", # This dependency introduces breaking changes in patch releases
69+
# For checking the noxfile, docs/ script, and tests
70+
"frequenz-sdk[dev-mkdocs,dev-noxfile,dev-pytest]",
5871
]
59-
format = ["black == 23.3.0", "isort == 5.12.0"]
60-
nox = ["nox == 2023.4.22", "toml == 0.10.2"]
61-
examples = [ "polars == 0.18.6" ]
62-
pytest = [
72+
dev-noxfile = ["nox == 2023.4.22", "frequenz-repo-config[lib] == 0.3.0"]
73+
dev-pylint = [
74+
"pylint == 2.17.4",
75+
# For checking the noxfile, docs/ script, and tests
76+
"frequenz-sdk[dev-mkdocs,dev-noxfile,dev-pytest]",
77+
]
78+
dev-pytest = [
6379
"pytest == 7.4.0",
64-
"pytest-cov == 4.1.0",
6580
"pytest-mock == 3.11.1",
6681
"pytest-asyncio == 0.21.0",
6782
"time-machine == 2.10.0",
6883
"async-solipsism == 0.5",
6984
# For checking docstring code examples
7085
"sybil == 5.0.2",
7186
"pylint == 2.17.4",
72-
"frequenz-sdk[examples]",
73-
]
74-
mypy = [
75-
"mypy == 1.4.1",
76-
"grpc-stubs == 1.24.12", # This dependency introduces breaking changes in patch releases
77-
# For checking the noxfile, docs/ script, and tests
78-
"frequenz-sdk[docs-gen,nox,pytest]",
87+
"frequenz-sdk[dev-examples]",
7988
]
80-
pylint = [
81-
"pylint == 2.17.4",
82-
# For checking the noxfile, docs/ script, and tests
83-
"frequenz-sdk[docs-gen,nox,pytest]",
89+
dev = [
90+
"frequenz-sdk[dev-mkdocs,dev-docstrings,dev-formatting,dev-mkdocs,dev-mypy,dev-noxfile,dev-pylint,dev-pytest]",
8491
]
85-
dev = ["frequenz-sdk[docs-gen,docs-lint,format,nox,pytest,mypy,pylint]"]
8692

8793
[project.urls]
8894
Changelog = "https://github.com/frequenz-floss/frequenz-sdk-python/releases"
89-
Repository = "https://github.com/frequenz-floss/frequenz-sdk-python"
9095
Issues = "https://github.com/frequenz-floss/frequenz-sdk-python/issues"
96+
Repository = "https://github.com/frequenz-floss/frequenz-sdk-python"
9197
Support = "https://github.com/frequenz-floss/frequenz-sdk-python/discussions/categories/support"
9298

93-
[tool.setuptools]
94-
include-package-data = true
95-
96-
[tool.setuptools_scm]
97-
version_scheme = "post-release"
98-
9999
[tool.black]
100100
line-length = 88
101-
target-version = ['py38']
101+
target-version = ['py311']
102102
include = '\.pyi?$'
103103

104+
[tool.isort]
105+
profile = "black"
106+
line_length = 88
107+
src_paths = ["src", "examples", "tests"]
108+
104109
[tool.pylint.similarities]
105110
ignore-comments = ['yes']
106111
ignore-docstrings = ['yes']
@@ -121,17 +126,15 @@ disable = [
121126
[tool.pylint.design]
122127
max-attributes = 12
123128

124-
[tool.isort]
125-
profile = "black"
126-
line_length = 88
127-
src_paths = ["src", "examples", "tests"]
128-
129129
[tool.pytest.ini_options]
130+
testpaths = ["tests", "src"]
130131
asyncio_mode = "auto"
131132
required_plugins = ["pytest-asyncio", "pytest-mock"]
132133

133134
[[tool.mypy.overrides]]
134135
module = [
136+
"async_solipsism",
137+
"async_solipsism.*",
135138
"grpc.aio",
136139
"grpc.aio.*",
137140
# There is a stubs package available, but it's not working:
@@ -140,6 +143,5 @@ module = [
140143
]
141144
ignore_missing_imports = true
142145

143-
[[tool.mypy.overrides]]
144-
module = ["async_solipsism", "async_solipsism.*"]
145-
ignore_missing_imports = true
146+
[tool.setuptools_scm]
147+
version_scheme = "post-release"

0 commit comments

Comments
 (0)
Please sign in to comment.