fastcs-eiger 0.4.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. fastcs_eiger-0.4.0/.copier-answers.yml +20 -0
  2. fastcs_eiger-0.4.0/.devcontainer/devcontainer.json +72 -0
  3. fastcs_eiger-0.4.0/.github/CONTRIBUTING.md +27 -0
  4. fastcs_eiger-0.4.0/.github/ISSUE_TEMPLATE/bug_report.md +21 -0
  5. fastcs_eiger-0.4.0/.github/ISSUE_TEMPLATE/issue.md +13 -0
  6. fastcs_eiger-0.4.0/.github/PULL_REQUEST_TEMPLATE/pull_request_template.md +8 -0
  7. fastcs_eiger-0.4.0/.github/pages/index.html +11 -0
  8. fastcs_eiger-0.4.0/.github/pages/make_switcher.py +96 -0
  9. fastcs_eiger-0.4.0/.github/workflows/_container.yml +64 -0
  10. fastcs_eiger-0.4.0/.github/workflows/_debug_container.yml +49 -0
  11. fastcs_eiger-0.4.0/.github/workflows/_dist.yml +37 -0
  12. fastcs_eiger-0.4.0/.github/workflows/_docs.yml +55 -0
  13. fastcs_eiger-0.4.0/.github/workflows/_pypi.yml +19 -0
  14. fastcs_eiger-0.4.0/.github/workflows/_release.yml +32 -0
  15. fastcs_eiger-0.4.0/.github/workflows/_test.yml +39 -0
  16. fastcs_eiger-0.4.0/.github/workflows/_tox.yml +22 -0
  17. fastcs_eiger-0.4.0/.github/workflows/ci.yml +67 -0
  18. fastcs_eiger-0.4.0/.github/workflows/periodic.yml +13 -0
  19. fastcs_eiger-0.4.0/.gitignore +74 -0
  20. fastcs_eiger-0.4.0/.gitleaks.toml +19 -0
  21. fastcs_eiger-0.4.0/.pre-commit-config.yaml +37 -0
  22. fastcs_eiger-0.4.0/.python-version +1 -0
  23. fastcs_eiger-0.4.0/.vscode/extensions.json +5 -0
  24. fastcs_eiger-0.4.0/.vscode/launch.json +54 -0
  25. fastcs_eiger-0.4.0/.vscode/settings.json +12 -0
  26. fastcs_eiger-0.4.0/.vscode/tasks.json +16 -0
  27. fastcs_eiger-0.4.0/Dockerfile +71 -0
  28. fastcs_eiger-0.4.0/LICENSE +201 -0
  29. fastcs_eiger-0.4.0/PKG-INFO +266 -0
  30. fastcs_eiger-0.4.0/README.md +42 -0
  31. fastcs_eiger-0.4.0/catalog-info.yaml +10 -0
  32. fastcs_eiger-0.4.0/docs/_api.rst +16 -0
  33. fastcs_eiger-0.4.0/docs/_templates/custom-module-template.rst +37 -0
  34. fastcs_eiger-0.4.0/docs/conf.py +191 -0
  35. fastcs_eiger-0.4.0/docs/explanations/decisions/0001-record-architecture-decisions.md +18 -0
  36. fastcs_eiger-0.4.0/docs/explanations/decisions/0002-switched-to-python-copier-template.md +28 -0
  37. fastcs_eiger-0.4.0/docs/explanations/decisions/COPYME +19 -0
  38. fastcs_eiger-0.4.0/docs/explanations/decisions.md +12 -0
  39. fastcs_eiger-0.4.0/docs/explanations.md +10 -0
  40. fastcs_eiger-0.4.0/docs/genindex.md +3 -0
  41. fastcs_eiger-0.4.0/docs/how-to/contribute.md +2 -0
  42. fastcs_eiger-0.4.0/docs/how-to/run-container.md +14 -0
  43. fastcs_eiger-0.4.0/docs/how-to.md +10 -0
  44. fastcs_eiger-0.4.0/docs/images/dls-logo.svg +11 -0
  45. fastcs_eiger-0.4.0/docs/index.md +56 -0
  46. fastcs_eiger-0.4.0/docs/reference.md +12 -0
  47. fastcs_eiger-0.4.0/docs/tutorials/installation.md +42 -0
  48. fastcs_eiger-0.4.0/docs/tutorials.md +10 -0
  49. fastcs_eiger-0.4.0/pyproject.toml +185 -0
  50. fastcs_eiger-0.4.0/renovate.json +41 -0
  51. fastcs_eiger-0.4.0/run_acquisition.py +107 -0
  52. fastcs_eiger-0.4.0/setup.cfg +4 -0
  53. fastcs_eiger-0.4.0/src/fastcs_eiger/__init__.py +3 -0
  54. fastcs_eiger-0.4.0/src/fastcs_eiger/__main__.py +88 -0
  55. fastcs_eiger-0.4.0/src/fastcs_eiger/_version.py +34 -0
  56. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/eiger_controller.py +146 -0
  57. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/eiger_detector_controller.py +60 -0
  58. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/eiger_monitor_controller.py +25 -0
  59. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/eiger_stream_controller.py +5 -0
  60. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/eiger_subsystem_controller.py +178 -0
  61. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/odin/eiger_fan.py +35 -0
  62. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/odin/eiger_fp_adapter_controller.py +11 -0
  63. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/odin/eiger_odin_controller.py +95 -0
  64. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/odin/generate_vds.py +120 -0
  65. fastcs_eiger-0.4.0/src/fastcs_eiger/controllers/odin/odin_controller.py +78 -0
  66. fastcs_eiger-0.4.0/src/fastcs_eiger/eiger_parameter.py +89 -0
  67. fastcs_eiger-0.4.0/src/fastcs_eiger/http_connection.py +115 -0
  68. fastcs_eiger-0.4.0/src/fastcs_eiger/io.py +85 -0
  69. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/PKG-INFO +266 -0
  70. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/SOURCES.txt +88 -0
  71. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/dependency_links.txt +1 -0
  72. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/entry_points.txt +2 -0
  73. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/requires.txt +6 -0
  74. fastcs_eiger-0.4.0/src/fastcs_eiger.egg-info/top_level.txt +1 -0
  75. fastcs_eiger-0.4.0/tests/conftest.py +48 -0
  76. fastcs_eiger-0.4.0/tests/input/ef_response.json +65 -0
  77. fastcs_eiger-0.4.0/tests/regenerate_test_output.sh +8 -0
  78. fastcs_eiger-0.4.0/tests/start_tickit_client.sh +3 -0
  79. fastcs_eiger-0.4.0/tests/system/eiger.yaml +3 -0
  80. fastcs_eiger-0.4.0/tests/system/parameters.json +939 -0
  81. fastcs_eiger-0.4.0/tests/system/test_eiger_introspection.py +360 -0
  82. fastcs_eiger-0.4.0/tests/test_cli.py +10 -0
  83. fastcs_eiger-0.4.0/tests/test_eiger_controller.py +149 -0
  84. fastcs_eiger-0.4.0/tests/test_eiger_fan_controller.py +56 -0
  85. fastcs_eiger-0.4.0/tests/test_eiger_odin_controller.py +115 -0
  86. fastcs_eiger-0.4.0/tests/test_eiger_parameter.py +27 -0
  87. fastcs_eiger-0.4.0/tests/test_generate_vds.py +358 -0
  88. fastcs_eiger-0.4.0/tests/test_io.py +21 -0
  89. fastcs_eiger-0.4.0/tests/test_odin_controller.py +37 -0
  90. fastcs_eiger-0.4.0/uv.lock +3805 -0
@@ -0,0 +1,20 @@
1
+ # Changes here will be overwritten by Copier
2
+ _commit: 5.0.2
3
+ _src_path: gh:DiamondLightSource/python-copier-template
4
+ author_email: gary.yendell@diamond.ac.uk
5
+ author_name: Gary Yendell
6
+ component_lifecycle: experimental
7
+ component_owner: user:mef65357
8
+ component_type: service
9
+ description: Eiger control system integration with FastCS
10
+ distribution_name: fastcs-eiger
11
+ docker: true
12
+ docker_debug: true
13
+ docs_type: sphinx
14
+ git_platform: github.com
15
+ github_org: DiamondLightSource
16
+ package_name: fastcs_eiger
17
+ pypi: true
18
+ repo_name: fastcs-eiger
19
+ strict_typing: false
20
+ type_checker: pyright
@@ -0,0 +1,72 @@
1
+ // For format details, see https://containers.dev/implementors/json_reference/
2
+ {
3
+ "name": "Python 3 Developer Container",
4
+ "build": {
5
+ "dockerfile": "../Dockerfile",
6
+ "target": "developer"
7
+ },
8
+ "remoteEnv": {
9
+ // Allow X11 apps to run inside the container
10
+ "DISPLAY": "${localEnv:DISPLAY}",
11
+ // Put things that allow it in the persistent cache
12
+ "PRE_COMMIT_HOME": "/cache/pre-commit",
13
+ "UV_CACHE_DIR": "/cache/uv",
14
+ "UV_PYTHON_CACHE_DIR": "/cache/uv-python",
15
+ // Make a venv that is specific for this workspace path as the cache is shared
16
+ "UV_PROJECT_ENVIRONMENT": "/cache/venv-for${localWorkspaceFolder}",
17
+ // Do the equivalent of "activate" the venv so we don't have to "uv run" everything
18
+ "VIRTUAL_ENV": "/cache/venv-for${localWorkspaceFolder}",
19
+ "PATH": "/cache/venv-for${localWorkspaceFolder}/bin:${containerEnv:PATH}"
20
+ },
21
+ "customizations": {
22
+ "vscode": {
23
+ // Set *default* container specific settings.json values on container create.
24
+ "settings": {
25
+ // Use the container's python by default
26
+ "python.defaultInterpreterPath": "/cache/venv-for${localWorkspaceFolder}/bin/python",
27
+ // Don't activate the venv as it is already in the PATH
28
+ "python.terminal.activateEnvInCurrentTerminal": false,
29
+ "python.terminal.activateEnvironment": false,
30
+ // Workaround to prevent garbled python REPL in the terminal
31
+ // https://github.com/microsoft/vscode-python/issues/25505
32
+ "python.terminal.shellIntegration.enabled": false
33
+ },
34
+ // Add the IDs of extensions you want installed when the container is created.
35
+ "extensions": [
36
+ "ms-python.python",
37
+ "github.vscode-github-actions",
38
+ "tamasfe.even-better-toml",
39
+ "redhat.vscode-yaml",
40
+ "ryanluker.vscode-coverage-gutters",
41
+ "charliermarsh.ruff",
42
+ "ms-azuretools.vscode-docker"
43
+ ]
44
+ }
45
+ },
46
+ // Create the config folder for the bash-config feature and uv cache
47
+ "initializeCommand": "mkdir -p ${localEnv:HOME}/.config/terminal-config",
48
+ "runArgs": [
49
+ // Allow the container to access the host X11 display and EPICS CA
50
+ "--net=host",
51
+ // Make sure SELinux does not disable with access to host filesystems like tmp
52
+ "--security-opt=label=disable"
53
+ ],
54
+ "mounts": [
55
+ // Mount in the user terminal config folder so it can be edited
56
+ {
57
+ "source": "${localEnv:HOME}/.config/terminal-config",
58
+ "target": "/user-terminal-config",
59
+ "type": "bind"
60
+ },
61
+ // Keep a persistent cross container cache for uv, pre-commit, and the venvs
62
+ {
63
+ "source": "devcontainer-shared-cache",
64
+ "target": "/cache",
65
+ "type": "volume"
66
+ }
67
+ ],
68
+ // Mount the parent as /workspaces so we can pip install peers as editable
69
+ "workspaceMount": "source=${localWorkspaceFolder}/..,target=/workspaces,type=bind",
70
+ // After the container is created, recreate the venv then make pre-commit first run faster
71
+ "postCreateCommand": "uv venv --clear && uv sync && pre-commit install --install-hooks"
72
+ }
@@ -0,0 +1,27 @@
1
+ # Contribute to the project
2
+
3
+ Contributions and issues are most welcome! All issues and pull requests are
4
+ handled through [GitHub](https://github.com/DiamondLightSource/fastcs-eiger/issues). Also, please check for any existing issues before
5
+ filing a new one. If you have a great idea but it involves big changes, please
6
+ file a ticket before making a pull request! We want to make sure you don't spend
7
+ your time coding something that might not fit the scope of the project.
8
+
9
+ ## Issue or Discussion?
10
+
11
+ Github also offers [discussions](https://github.com/DiamondLightSource/fastcs-eiger/discussions) as a place to ask questions and share ideas. If
12
+ your issue is open ended and it is not obvious when it can be "closed", please
13
+ raise it as a discussion instead.
14
+
15
+ ## Code Coverage
16
+
17
+ While 100% code coverage does not make a library bug-free, it significantly
18
+ reduces the number of easily caught bugs! Please make sure coverage remains the
19
+ same or is improved by a pull request!
20
+
21
+ ## Developer Information
22
+
23
+ It is recommended that developers use a [vscode devcontainer](https://code.visualstudio.com/docs/devcontainers/containers). This repository contains configuration to set up a containerized development environment that suits its own needs.
24
+
25
+ This project was created using the [Diamond Light Source Copier Template](https://github.com/DiamondLightSource/python-copier-template) for Python projects.
26
+
27
+ For more information on common tasks like setting up a developer environment, running the tests, and setting a pre-commit hook, see the template's [How-to guides](https://diamondlightsource.github.io/python-copier-template/5.0.2/how-to.html).
@@ -0,0 +1,21 @@
1
+ ---
2
+ name: Bug Report
3
+ about: The template to use for reporting bugs and usability issues
4
+ title: " "
5
+ labels: 'bug'
6
+ assignees: ''
7
+
8
+ ---
9
+
10
+ Describe the bug, including a clear and concise description of the expected behaviour, the actual behavior and the context in which you encountered it (ideally include details of your environment).
11
+
12
+ ## Steps To Reproduce
13
+ Steps to reproduce the behavior:
14
+ 1. Go to '...'
15
+ 2. Click on '....'
16
+ 3. Scroll down to '....'
17
+ 4. See error
18
+
19
+
20
+ ## Acceptance Criteria
21
+ - Specific criteria that will be used to judge if the issue is fixed
@@ -0,0 +1,13 @@
1
+ ---
2
+ name: Issue
3
+ about: The standard template to use for feature requests, design discussions and tasks
4
+ title: " "
5
+ labels: ''
6
+ assignees: ''
7
+
8
+ ---
9
+
10
+ A brief description of the issue, including specific stakeholders and the business case where appropriate
11
+
12
+ ## Acceptance Criteria
13
+ - Specific criteria that will be used to judge if the issue is fixed
@@ -0,0 +1,8 @@
1
+ Fixes #ISSUE
2
+
3
+ ### Instructions to reviewer on how to test:
4
+ 1. Do thing x
5
+ 2. Confirm thing y happens
6
+
7
+ ### Checks for reviewer
8
+ - [ ] Would the PR title make sense to a user on a set of release notes
@@ -0,0 +1,11 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+
4
+ <head>
5
+ <title>Redirecting to main branch</title>
6
+ <meta charset="utf-8">
7
+ <meta http-equiv="refresh" content="0; url=./main/index.html">
8
+ <link rel="canonical" href="main/index.html">
9
+ </head>
10
+
11
+ </html>
@@ -0,0 +1,96 @@
1
+ """Make switcher.json to allow docs to switch between different versions."""
2
+
3
+ import json
4
+ import logging
5
+ from argparse import ArgumentParser
6
+ from pathlib import Path
7
+ from subprocess import CalledProcessError, check_output
8
+
9
+
10
+ def report_output(stdout: bytes, label: str) -> list[str]:
11
+ """Print and return something received frm stdout."""
12
+ ret = stdout.decode().strip().split("\n")
13
+ print(f"{label}: {ret}")
14
+ return ret
15
+
16
+
17
+ def get_branch_contents(ref: str) -> list[str]:
18
+ """Get the list of directories in a branch."""
19
+ stdout = check_output(["git", "ls-tree", "-d", "--name-only", ref])
20
+ return report_output(stdout, "Branch contents")
21
+
22
+
23
+ def get_sorted_tags_list() -> list[str]:
24
+ """Get a list of sorted tags in descending order from the repository."""
25
+ stdout = check_output(["git", "tag", "-l", "--sort=-v:refname"])
26
+ return report_output(stdout, "Tags list")
27
+
28
+
29
+ def get_versions(ref: str, add: str | None) -> list[str]:
30
+ """Generate the file containing the list of all GitHub Pages builds."""
31
+ # Get the directories (i.e. builds) from the GitHub Pages branch
32
+ try:
33
+ builds = set(get_branch_contents(ref))
34
+ except CalledProcessError:
35
+ builds = set()
36
+ logging.warning(f"Cannot get {ref} contents")
37
+
38
+ # Add and remove from the list of builds
39
+ if add:
40
+ builds.add(add)
41
+
42
+ # Get a sorted list of tags
43
+ tags = get_sorted_tags_list()
44
+
45
+ # Make the sorted versions list from main branches and tags
46
+ versions: list[str] = []
47
+ for version in ["master", "main"] + tags:
48
+ if version in builds:
49
+ versions.append(version)
50
+ builds.remove(version)
51
+
52
+ # Add in anything that is left to the bottom
53
+ versions += sorted(builds)
54
+ print(f"Sorted versions: {versions}")
55
+ return versions
56
+
57
+
58
+ def write_json(path: Path, repository: str, versions: list[str]):
59
+ """Write the JSON switcher to path."""
60
+ org, repo_name = repository.split("/")
61
+ struct = [
62
+ {"version": version, "url": f"https://{org}.github.io/{repo_name}/{version}/"}
63
+ for version in versions
64
+ ]
65
+ text = json.dumps(struct, indent=2)
66
+ print(f"JSON switcher:\n{text}")
67
+ path.write_text(text, encoding="utf-8")
68
+
69
+
70
+ def main(args=None):
71
+ """Parse args and write switcher."""
72
+ parser = ArgumentParser(
73
+ description="Make a versions.json file from gh-pages directories"
74
+ )
75
+ parser.add_argument(
76
+ "--add",
77
+ help="Add this directory to the list of existing directories",
78
+ )
79
+ parser.add_argument(
80
+ "repository",
81
+ help="The GitHub org and repository name: ORG/REPO",
82
+ )
83
+ parser.add_argument(
84
+ "output",
85
+ type=Path,
86
+ help="Path of write switcher.json to",
87
+ )
88
+ args = parser.parse_args(args)
89
+
90
+ # Write the versions file
91
+ versions = get_versions("origin/gh-pages", args.add)
92
+ write_json(args.output, args.repository, versions)
93
+
94
+
95
+ if __name__ == "__main__":
96
+ main()
@@ -0,0 +1,64 @@
1
+ on:
2
+ workflow_call:
3
+ inputs:
4
+ publish:
5
+ type: boolean
6
+ description: If true, pushes image to container registry
7
+
8
+ jobs:
9
+ build:
10
+ runs-on: ubuntu-latest
11
+
12
+ steps:
13
+ - name: Checkout
14
+ uses: actions/checkout@v5
15
+ with:
16
+ # Need this to get version number from last tag
17
+ fetch-depth: 0
18
+
19
+ - name: Set up Docker Buildx
20
+ id: buildx
21
+ uses: docker/setup-buildx-action@v3
22
+
23
+ - name: Log in to GitHub Docker Registry
24
+ if: github.event_name != 'pull_request'
25
+ uses: docker/login-action@v3
26
+ with:
27
+ registry: ghcr.io
28
+ username: ${{ github.actor }}
29
+ password: ${{ secrets.GITHUB_TOKEN }}
30
+
31
+ - name: Build and export to Docker local cache
32
+ uses: docker/build-push-action@v6
33
+ env:
34
+ DOCKER_BUILD_RECORD_UPLOAD: false
35
+ with:
36
+ context: .
37
+ # Need load and tags so we can test it below
38
+ load: true
39
+ tags: tag_for_testing
40
+
41
+ - name: Test cli works in cached runtime image
42
+ run: docker run --rm tag_for_testing --version
43
+
44
+ - name: Create tags for publishing image
45
+ id: meta
46
+ uses: docker/metadata-action@v5
47
+ with:
48
+ images: ghcr.io/${{ github.repository }}
49
+ tags: |
50
+ type=ref,event=tag
51
+ type=raw,value=latest
52
+
53
+ - name: Push cached image to container registry
54
+ if: inputs.publish && github.ref_type == 'tag'
55
+ uses: docker/build-push-action@v6
56
+ env:
57
+ DOCKER_BUILD_RECORD_UPLOAD: false
58
+ # This does not build the image again, it will find the image in the
59
+ # Docker cache and publish it
60
+ with:
61
+ context: .
62
+ push: true
63
+ tags: ${{ steps.meta.outputs.tags }}
64
+ labels: ${{ steps.meta.outputs.labels }}
@@ -0,0 +1,49 @@
1
+ on:
2
+ workflow_call:
3
+ inputs:
4
+ publish:
5
+ type: boolean
6
+ description: If true, pushes image to container registry
7
+
8
+ jobs:
9
+ build:
10
+ runs-on: ubuntu-latest
11
+
12
+ steps:
13
+ - name: Checkout
14
+ uses: actions/checkout@v4
15
+ with:
16
+ # Need this to get version number from last tag
17
+ fetch-depth: 0
18
+
19
+ - name: Set up Docker Buildx
20
+ id: buildx
21
+ uses: docker/setup-buildx-action@v3
22
+
23
+ - name: Log in to GitHub Docker Registry
24
+ if: github.event_name != 'pull_request'
25
+ uses: docker/login-action@v3
26
+ with:
27
+ registry: ghcr.io
28
+ username: ${{ github.actor }}
29
+ password: ${{ secrets.GITHUB_TOKEN }}
30
+
31
+ - name: Create tags for publishing debug image
32
+ id: debug-meta
33
+ uses: docker/metadata-action@v5
34
+ with:
35
+ images: ghcr.io/${{ github.repository }}
36
+ tags: |
37
+ type=ref,event=tag,suffix=-debug
38
+ type=raw,value=latest-debug
39
+
40
+ - name: Build and publish debug image to container registry
41
+ if: github.ref_type == 'tag'
42
+ uses: docker/build-push-action@v6
43
+ env:
44
+ DOCKER_BUILD_RECORD_UPLOAD: false
45
+ with:
46
+ context: .
47
+ push: true
48
+ target: debug
49
+ tags: ${{ steps.debug-meta.outputs.tags }}
@@ -0,0 +1,37 @@
1
+ on:
2
+ workflow_call:
3
+
4
+ jobs:
5
+ build:
6
+ runs-on: "ubuntu-latest"
7
+
8
+ steps:
9
+ - name: Checkout
10
+ uses: actions/checkout@v5
11
+ with:
12
+ # Need this to get version number from last tag
13
+ fetch-depth: 0
14
+
15
+ - name: Install uv
16
+ uses: astral-sh/setup-uv@v7
17
+
18
+ - name: Build sdist and wheel
19
+ run: >
20
+ export SOURCE_DATE_EPOCH=$(git log -1 --pretty=%ct) &&
21
+ uvx --from build pyproject-build
22
+
23
+ - name: Upload sdist and wheel as artifacts
24
+ uses: actions/upload-artifact@v4
25
+ with:
26
+ name: dist
27
+ path: dist
28
+
29
+ - name: Check for packaging errors
30
+ run: uvx twine check --strict dist/*
31
+
32
+ - name: Install produced wheel
33
+ run: python -m pip install dist/*.whl
34
+
35
+ - name: Test module --version works using the installed wheel
36
+ # If more than one module in src/ replace with module name to test
37
+ run: python -m $(ls --hide='*.egg-info' src | head -1) --version
@@ -0,0 +1,55 @@
1
+ on:
2
+ workflow_call:
3
+
4
+
5
+ jobs:
6
+ build:
7
+ runs-on: ubuntu-latest
8
+
9
+ steps:
10
+ - name: Avoid git conflicts when tag and branch pushed at same time
11
+ if: github.ref_type == 'tag'
12
+ run: sleep 60
13
+
14
+ - name: Checkout
15
+ uses: actions/checkout@v5
16
+ with:
17
+ # Need this to get version number from last tag
18
+ fetch-depth: 0
19
+
20
+ - name: Install system packages
21
+ run: sudo apt-get install graphviz
22
+
23
+ - name: Install uv
24
+ uses: astral-sh/setup-uv@v7
25
+
26
+ - name: Build docs
27
+ run: uv run --locked tox -e docs
28
+
29
+ - name: Remove environment.pickle
30
+ run: rm build/html/.doctrees/environment.pickle
31
+
32
+ - name: Upload built docs artifact
33
+ uses: actions/upload-artifact@v4
34
+ with:
35
+ name: docs
36
+ path: build
37
+
38
+ - name: Sanitize ref name for docs version
39
+ run: echo "DOCS_VERSION=${GITHUB_REF_NAME//[^A-Za-z0-9._-]/_}" >> $GITHUB_ENV
40
+
41
+ - name: Move to versioned directory
42
+ run: mv build/html .github/pages/$DOCS_VERSION
43
+
44
+ - name: Write switcher.json
45
+ run: python .github/pages/make_switcher.py --add $DOCS_VERSION ${{ github.repository }} .github/pages/switcher.json
46
+
47
+ - name: Publish Docs to gh-pages
48
+ if: github.ref_type == 'tag' || github.ref_name == 'main'
49
+ # We pin to the SHA, not the tag, for security reasons.
50
+ # https://docs.github.com/en/actions/learn-github-actions/security-hardening-for-github-actions#using-third-party-actions
51
+ uses: peaceiris/actions-gh-pages@4f9cc6602d3f66b9c108549d475ec49e8ef4d45e # v4.0.0
52
+ with:
53
+ github_token: ${{ secrets.GITHUB_TOKEN }}
54
+ publish_dir: .github/pages
55
+ keep_files: true
@@ -0,0 +1,19 @@
1
+ on:
2
+ workflow_call:
3
+
4
+ jobs:
5
+ upload:
6
+ runs-on: ubuntu-latest
7
+ environment: release
8
+
9
+ steps:
10
+ - name: Download dist artifact
11
+ uses: actions/download-artifact@v5
12
+ with:
13
+ name: dist
14
+ path: dist
15
+
16
+ - name: Publish to PyPI using trusted publishing
17
+ uses: pypa/gh-action-pypi-publish@release/v1
18
+ with:
19
+ attestations: false
@@ -0,0 +1,32 @@
1
+ on:
2
+ workflow_call:
3
+
4
+ jobs:
5
+ artifacts:
6
+ runs-on: ubuntu-latest
7
+
8
+ steps:
9
+ - name: Download artifacts
10
+ uses: actions/download-artifact@v5
11
+ with:
12
+ merge-multiple: true
13
+
14
+ - name: Zip up docs
15
+ run: |
16
+ set -vxeuo pipefail
17
+ if [ -d html ]; then
18
+ mv html $GITHUB_REF_NAME
19
+ zip -r docs.zip $GITHUB_REF_NAME
20
+ rm -rf $GITHUB_REF_NAME
21
+ fi
22
+
23
+ - name: Create GitHub Release
24
+ # We pin to the SHA, not the tag, for security reasons.
25
+ # https://docs.github.com/en/actions/learn-github-actions/security-hardening-for-github-actions#using-third-party-actions
26
+ uses: softprops/action-gh-release@6da8fa9354ddfdc4aeace5fc48d7f679b5214090 # v2.4.1
27
+ with:
28
+ prerelease: ${{ contains(github.ref_name, 'a') || contains(github.ref_name, 'b') || contains(github.ref_name, 'rc') }}
29
+ files: "*"
30
+ generate_release_notes: true
31
+ env:
32
+ GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
@@ -0,0 +1,39 @@
1
+ on:
2
+ workflow_call:
3
+ inputs:
4
+ python-version:
5
+ type: string
6
+ description: The version of python to install, default is from .python-version file
7
+ default: ""
8
+ runs-on:
9
+ type: string
10
+ description: The runner to run this job on
11
+ required: true
12
+
13
+ env:
14
+ # https://github.com/pytest-dev/pytest/issues/2042
15
+ PY_IGNORE_IMPORTMISMATCH: "1"
16
+ UV_PYTHON: ${{ inputs.python-version }}
17
+
18
+ jobs:
19
+ run:
20
+ runs-on: ${{ inputs.runs-on }}
21
+
22
+ steps:
23
+ - name: Checkout
24
+ uses: actions/checkout@v5
25
+ with:
26
+ # Need this to get version number from last tag
27
+ fetch-depth: 0
28
+
29
+ - name: Install uv
30
+ uses: astral-sh/setup-uv@v7
31
+
32
+ - name: Run tests
33
+ run: uv run --locked tox -e tests
34
+
35
+ - name: Upload coverage to Codecov
36
+ uses: codecov/codecov-action@v5
37
+ with:
38
+ name: ${{ inputs.python-version }}/${{ inputs.runs-on }}
39
+ files: cov.xml
@@ -0,0 +1,22 @@
1
+ on:
2
+ workflow_call:
3
+ inputs:
4
+ tox:
5
+ type: string
6
+ description: What to run under tox
7
+ required: true
8
+
9
+
10
+ jobs:
11
+ run:
12
+ runs-on: "ubuntu-latest"
13
+
14
+ steps:
15
+ - name: Checkout
16
+ uses: actions/checkout@v5
17
+
18
+ - name: Install uv
19
+ uses: astral-sh/setup-uv@v7
20
+
21
+ - name: Run tox
22
+ run: uv run --locked tox -e ${{ inputs.tox }}
@@ -0,0 +1,67 @@
1
+ name: CI
2
+
3
+ on:
4
+ push:
5
+ branches:
6
+ - main
7
+ tags:
8
+ - "*"
9
+ pull_request:
10
+
11
+ jobs:
12
+ lint:
13
+ uses: ./.github/workflows/_tox.yml
14
+ with:
15
+ tox: pre-commit,type-checking
16
+
17
+ test:
18
+ strategy:
19
+ matrix:
20
+ runs-on: ["ubuntu-latest"] # can add windows-latest, macos-latest
21
+ python-version: ["3.11", "3.12", "3.13"]
22
+ fail-fast: false
23
+ uses: ./.github/workflows/_test.yml
24
+ with:
25
+ runs-on: ${{ matrix.runs-on }}
26
+ python-version: ${{ matrix.python-version }}
27
+
28
+ container:
29
+ needs: test
30
+ if: always()
31
+ uses: ./.github/workflows/_container.yml
32
+ with:
33
+ publish: ${{ needs.test.result == 'success' }}
34
+ permissions:
35
+ contents: read
36
+ packages: write
37
+
38
+ debug_container:
39
+ needs: [container, test]
40
+ uses: ./.github/workflows/_debug_container.yml
41
+ with:
42
+ publish: ${{ needs.test.result == 'success' }}
43
+ permissions:
44
+ contents: read
45
+ packages: write
46
+
47
+ docs:
48
+ uses: ./.github/workflows/_docs.yml
49
+ permissions:
50
+ contents: write
51
+
52
+ dist:
53
+ uses: ./.github/workflows/_dist.yml
54
+
55
+ pypi:
56
+ needs: [dist, test]
57
+ if: github.ref_type == 'tag'
58
+ uses: ./.github/workflows/_pypi.yml
59
+ permissions:
60
+ id-token: write
61
+
62
+ release:
63
+ needs: [dist, test, docs]
64
+ if: github.ref_type == 'tag'
65
+ uses: ./.github/workflows/_release.yml
66
+ permissions:
67
+ contents: write
@@ -0,0 +1,13 @@
1
+ name: Periodic
2
+
3
+ on:
4
+ workflow_dispatch:
5
+ schedule:
6
+ # Run weekly to check URL links still resolve
7
+ - cron: "0 8 * * WED"
8
+
9
+ jobs:
10
+ linkcheck:
11
+ uses: ./.github/workflows/_tox.yml
12
+ with:
13
+ tox: docs -- -b linkcheck