{"id":49718858,"url":"https://github.com/frenck/python-fumis","last_synced_at":"2026-05-08T22:14:34.209Z","repository":{"id":36305786,"uuid":"223260725","full_name":"frenck/python-fumis","owner":"frenck","description":"Asynchronous Python client for the Fumis WiRCU API","archived":false,"fork":false,"pushed_at":"2026-05-02T05:30:12.000Z","size":633,"stargazers_count":8,"open_issues_count":2,"forks_count":4,"subscribers_count":2,"default_branch":"main","last_synced_at":"2026-05-02T07:25:24.376Z","etag":null,"topics":["api-client","asynchronous","austroflamm","eco-spar","fumis","furnis","heta","iot","pellet-stove","rika","smart-home","wircu"],"latest_commit_sha":null,"homepage":"","language":"Python","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/frenck.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":".github/CONTRIBUTING.md","funding":".github/FUNDING.yml","license":"LICENSE.md","code_of_conduct":".github/CODE_OF_CONDUCT.md","threat_model":null,"audit":null,"citation":null,"codeowners":".github/CODEOWNERS","security":".github/SECURITY.md","support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null,"zenodo":null,"notice":null,"maintainers":null,"copyright":null,"agents":null,"dco":null,"cla":null},"funding":{"github":"frenck","patreon":"frenck","custom":"https://frenck.dev/donate/"}},"created_at":"2019-11-21T20:35:43.000Z","updated_at":"2026-05-02T05:30:15.000Z","dependencies_parsed_at":"2023-02-15T06:05:27.086Z","dependency_job_id":"d4b9f8a6-832a-42af-a389-c4f25e233027","html_url":"https://github.com/frenck/python-fumis","commit_stats":null,"previous_names":[],"tags_count":5,"template":false,"template_full_name":null,"purl":"pkg:github/frenck/python-fumis","repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/frenck%2Fpython-fumis","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/frenck%2Fpython-fumis/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/frenck%2Fpython-fumis/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/frenck%2Fpython-fumis/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/frenck","download_url":"https://codeload.github.com/frenck/python-fumis/tar.gz/refs/heads/main","sbom_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/frenck%2Fpython-fumis/sbom","scorecard":{"id":1246487,"data":{"date":"2026-04-24T19:38:11Z","repo":{"name":"github.com/frenck/python-fumis","commit":"39d1acb89c6f453257c5bbebc5544f201fa1eed4"},"scorecard":{"version":"v5.3.0","commit":"c22063e786c11f9dd714d777a687ff7c4599b600"},"score":7.3,"checks":[{"name":"Code-Review","score":0,"reason":"Found 0/9 approved changesets -- score normalized to 0","details":null,"documentation":{"short":"Determines if the project requires human code review before pull requests (aka merge requests) are merged.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#code-review"}},{"name":"Maintained","score":10,"reason":"17 commit(s) and 0 issue activity found in the last 90 days -- score normalized to 10","details":null,"documentation":{"short":"Determines if the project is \"actively maintained\".","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#maintained"}},{"name":"Dependency-Update-Tool","score":10,"reason":"update tool detected","details":["Info: detected update tool: RenovateBot: .github/renovate.json:1"],"documentation":{"short":"Determines if the project uses a dependency update tool.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#dependency-update-tool"}},{"name":"Binary-Artifacts","score":10,"reason":"no binaries found in the repo","details":null,"documentation":{"short":"Determines if the project has generated executable (binary) artifacts in the source repository.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#binary-artifacts"}},{"name":"Dangerous-Workflow","score":10,"reason":"no dangerous workflow patterns detected","details":null,"documentation":{"short":"Determines if the project's GitHub Action workflows avoid dangerous patterns.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#dangerous-workflow"}},{"name":"Token-Permissions","score":10,"reason":"GitHub workflow tokens follow principle of least privilege","details":["Info: jobLevel 'contents' permission set to 'read': .github/workflows/labels.yaml:21","Info: jobLevel 'pull-requests' permission set to 'read': .github/workflows/pr-labels.yaml:23","Warn: jobLevel 'contents' permission set to 'write': .github/workflows/release-drafter.yaml:19","Warn: jobLevel 'contents' permission set to 'write': .github/workflows/release.yaml:25","Info: topLevel 'contents' permission set to 'read': .github/workflows/dependency-review.yaml:8","Info: topLevel 'contents' permission set to 'read': .github/workflows/labels.yaml:14","Info: topLevel 'contents' permission set to 'read': .github/workflows/linting.yaml:14","Info: topLevel 'contents' permission set to 'read': .github/workflows/lock.yaml:11","Info: topLevel 'contents' permission set to 'read': .github/workflows/pr-labels.yaml:16","Info: topLevel 'contents' permission set to 'read': .github/workflows/release-drafter.yaml:12","Info: topLevel 'contents' permission set to 'read': .github/workflows/release.yaml:14","Info: topLevel permissions set to 'read-all': .github/workflows/scorecard.yml:16","Info: topLevel 'contents' permission set to 'read': .github/workflows/tests.yaml:14","Info: topLevel 'contents' permission set to 'read': .github/workflows/typing.yaml:14"],"documentation":{"short":"Determines if the project's workflows follow the principle of least privilege.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#token-permissions"}},{"name":"Security-Policy","score":10,"reason":"security policy file detected","details":["Info: security policy file detected: .github/SECURITY.md:1","Info: Found linked content: .github/SECURITY.md:1","Info: Found disclosure, vulnerability, and/or timelines in security policy: .github/SECURITY.md:1","Info: Found text in security policy: .github/SECURITY.md:1"],"documentation":{"short":"Determines if the project has published a security policy.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#security-policy"}},{"name":"Pinned-Dependencies","score":9,"reason":"dependency not pinned by hash detected -- score normalized to 9","details":["Warn: npmCommand not pinned by hash: .github/workflows/linting.yaml:195","Info:  32 out of  32 GitHub-owned GitHubAction dependencies pinned","Info:   8 out of   8 third-party GitHubAction dependencies pinned","Info:   0 out of   1 npmCommand dependencies pinned"],"documentation":{"short":"Determines if the project has declared and pinned the dependencies of its build process.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#pinned-dependencies"}},{"name":"License","score":10,"reason":"license file detected","details":["Info: project has a license file: LICENSE.md:0","Info: FSF or OSI recognized license: MIT License: LICENSE.md:0"],"documentation":{"short":"Determines if the project has defined a license.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#license"}},{"name":"Vulnerabilities","score":10,"reason":"0 existing vulnerabilities detected","details":null,"documentation":{"short":"Determines if the project has open, known unfixed vulnerabilities.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#vulnerabilities"}},{"name":"CII-Best-Practices","score":0,"reason":"no effort to earn an OpenSSF best practices badge detected","details":null,"documentation":{"short":"Determines if the project has an OpenSSF (formerly CII) Best Practices Badge.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#cii-best-practices"}},{"name":"SAST","score":0,"reason":"SAST tool is not run on all commits -- score normalized to 0","details":["Warn: 0 commits out of 30 are checked with a SAST tool"],"documentation":{"short":"Determines if the project uses static code analysis.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#sast"}},{"name":"Packaging","score":10,"reason":"packaging workflow detected","details":["Info: Project packages its releases by way of GitHub Actions.: .github/workflows/release.yaml:17"],"documentation":{"short":"Determines if the project is published as a package that others can easily download, install, easily update, and uninstall.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#packaging"}},{"name":"Branch-Protection","score":4,"reason":"branch protection is not maximal on development and all release branches","details":["Info: 'allow deletion' disabled on branch 'main'","Info: 'force pushes' disabled on branch 'main'","Warn: 'branch protection settings apply to administrators' is disabled on branch 'main'","Info: 'stale review dismissal' is required to merge on branch 'main'","Warn: required approving review count is 1 on branch 'main'","Info: codeowner review is required on branch 'main'","Warn: 'last push approval' is disabled on branch 'main'","Warn: 'up-to-date branches' is disabled on branch 'main'","Info: status check found to merge onto on branch 'main'","Info: PRs are required in order to make changes on branch 'main'"],"documentation":{"short":"Determines if the default and release branches are protected with GitHub's branch protection settings.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#branch-protection"}},{"name":"Signed-Releases","score":-1,"reason":"no releases found","details":null,"documentation":{"short":"Determines if the project cryptographically signs release artifacts.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#signed-releases"}},{"name":"Fuzzing","score":0,"reason":"project is not fuzzed","details":["Warn: no fuzzer integrations found"],"documentation":{"short":"Determines if the project uses fuzzing.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#fuzzing"}},{"name":"Contributors","score":10,"reason":"project has 7 contributing companies or organizations","details":["Info: found contributions from: GitHub-Stars, esphome, hacs, hassio-addons, home-assistant, home-assistant @hassio-addons, home-assistant-libs"],"documentation":{"short":"Determines if the project has a set of contributors from multiple organizations (e.g., companies).","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#contributors"}},{"name":"CI-Tests","score":5,"reason":"17 out of 30 merged PRs checked by a CI test -- score normalized to 5","details":null,"documentation":{"short":"Determines if the project runs tests before pull requests are merged.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#ci-tests"}}]},"last_synced_at":"2026-04-24T21:32:01.301Z","repository_id":36305786,"created_at":"2026-04-24T21:32:01.301Z","updated_at":"2026-04-24T21:32:01.301Z"},"host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":286080680,"owners_count":32579084,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2026-05-03T06:36:36.687Z","status":"ssl_error","status_checked_at":"2026-05-03T06:36:09.306Z","response_time":103,"last_error":"SSL_read: unexpected eof while reading","robots_txt_status":"success","robots_txt_updated_at":"2025-07-24T06:49:26.215Z","robots_txt_url":"https://github.com/robots.txt","online":false,"can_crawl_api":true,"host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["api-client","asynchronous","austroflamm","eco-spar","fumis","furnis","heta","iot","pellet-stove","rika","smart-home","wircu"],"created_at":"2026-05-08T22:14:32.063Z","updated_at":"2026-05-08T22:14:34.201Z","avatar_url":"https://github.com/frenck.png","language":"Python","funding_links":["https://github.com/sponsors/frenck","https://patreon.com/frenck","https://frenck.dev/donate/","https://www.patreon.com/frenck"],"categories":[],"sub_categories":[],"readme":"# Python: Asynchronous client for the Fumis WiRCU API\n\n[![GitHub Release][releases-shield]][releases]\n[![Python Versions][python-versions-shield]][pypi]\n![Project Stage][project-stage-shield]\n![Project Maintenance][maintenance-shield]\n[![License][license-shield]](LICENSE.md)\n\n[![Build Status][build-shield]][build]\n[![Code Coverage][codecov-shield]][codecov]\n[![OpenSSF Scorecard][scorecard-shield]][scorecard]\n[![Open in Dev Containers][devcontainer-shield]][devcontainer]\n\n[![Sponsor Frenck via GitHub Sponsors][github-sponsors-shield]][github-sponsors]\n\n[![Support Frenck on Patreon][patreon-shield]][patreon]\n\nAsynchronous Python client for the Fumis WiRCU API.\n\n## About\n\nThis package allows you to control and monitor Fumis WiRCU pellet stove\ndevices programmatically. It is mainly created to allow third-party programs\nto automate the behavior of a Fumis WiRCU device.\n\nAn excellent example of this might be Home Assistant, which allows you to write\nautomations, to turn on your pellet stove and set a target temperature.\n\nKnown compatible stove brands:\n\n- Austroflamm (Clou Duo, MO DUO, Polly 2.0)\n- Heta (Green 200)\n- HAAS+SOHN\n- Eco Spar (Auriga, Solara, Tukana, Karina, Nova)\n- Prity\n\n## Installation\n\n```bash\npip install fumis\n```\n\nTo install with the optional CLI:\n\n```bash\npip install \"fumis[cli]\"\n```\n\n## CLI\n\nThe optional CLI lets you control your stove directly from the terminal.\nThe `--mac` and `--password` options can also be set via the `FUMIS_MAC`\nand `FUMIS_PASSWORD` environment variables.\n\n```bash\n# Set credentials once via environment variables\nexport FUMIS_MAC=AABBCCDDEEFF\nexport FUMIS_PASSWORD=1234\n\n# Launch the live TUI dashboard\nfumis\n\n# Show device information\nfumis info\n\n# Turn the stove on/off\nfumis on\nfumis off\n\n# Set target temperature\nfumis temperature 23.5\n\n# Set power level (1-5)\nfumis power 3\n\n# Enable/disable eco mode\nfumis eco true\n\n# Show weekly timer schedule / enable / disable\nfumis timer\nfumis timer true\n\n# Sync the stove's clock to your system time\nfumis sync-clock\n\n# Show service diagnostics (sensors, IO, temperature channels)\nfumis diagnostics\n\n# Dump raw API response as JSON\nfumis dump\n\n# Emit machine-readable JSON\nfumis info --json\n```\n\nThe default command (no subcommand) launches a live TUI dashboard with\nreal-time temperature graphs, status display, and keyboard controls for\non/off, temperature, and power level.\n\n## Usage\n\nThe client is an async context manager; every API call is a coroutine:\n\n```python\nimport asyncio\n\nfrom fumis import Fumis, StoveStatus\n\n\nasync def main() -\u003e None:\n    \"\"\"Show example on controlling your Fumis WiRCU device.\"\"\"\n    async with Fumis(mac=\"AABBCCDDEEFF\", password=\"1234\") as fumis:\n        info = await fumis.update_info()\n\n        # Stove identity\n        print(info.controller.manufacturer)  # \"Austroflamm\"\n        print(info.controller.model_name)    # \"Clou Duo\"\n\n        # Status\n        print(info.controller.stove_status)  # StoveStatus.OFF\n        print(info.controller.on)            # False\n\n        # Temperatures\n        main_temp = info.controller.main_temperature\n        if main_temp:\n            print(f\"Room: {main_temp.actual}° → {main_temp.setpoint}°\")\n\n        # Combustion chamber\n        print(info.controller.combustion_chamber_temperature)\n\n        # Power\n        print(f\"{info.controller.power.kw} kW (level {info.controller.power.set_power})\")\n\n        # Door sensor\n        print(f\"Door open: {info.controller.door_open}\")\n\n        # Fuel level\n        fuel = info.controller.fuel()\n        if fuel:\n            print(f\"Fuel: {fuel.quantity_percentage}%\")\n\n        # Weekly timer schedule\n        schedule = info.controller.schedule\n        print(f\"Timer: {'on' if info.controller.timer_enable else 'off'}\")\n        for prog in schedule.programs:\n            if prog.active:\n                print(f\"  {prog}\")  # \"21:00-22:10\"\n        print(f\"Active days: {schedule.active_days}\")\n\n        # Control the stove\n        await fumis.turn_on()\n        await fumis.set_target_temperature(23.0)\n        await fumis.set_power(3)\n        await fumis.set_eco_mode(enabled=True)\n        await fumis.set_timer(enabled=True)\n        await fumis.set_fan_speed(3)\n        await fumis.set_clock()\n        await fumis.turn_off()\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n```\n\n### Raw API access\n\nFor diagnostics, debugging, or fixture capture (useful for Home Assistant\ndiagnostic downloads):\n\n```python\nasync with Fumis(mac=\"AABBCCDDEEFF\", password=\"1234\") as fumis:\n    # Get the raw, unprocessed JSON dict from the API\n    raw = await fumis.raw_status()\n    print(json.dumps(raw, indent=2))\n```\n\n### Accessing diagnostic data\n\nAll diagnostic variables and parameters from the Fumis controller are\naccessible via the structured model:\n\n```python\ninfo = await fumis.update_info()\nc = info.controller\n\n# Convenience properties (return None if not available)\nc.exhaust_temperature     # Exhaust gas temp (var[11])\nc.fan1_speed              # Fan 1 speed (var[4])\nc.fan2_speed              # Fan 2 speed (var[12])\nc.f02                     # F02 sensor input (var[34])\nc.pressure                # Pressure sensor (var[35])\nc.door_open               # Door sensor (var[33]) - True/False/None\nc.stove_model             # Stove model ID (var[96])\nc.parameter_version       # Parameter version (var[97])\n\n# Raw diagnostic access (escape hatch)\nc.diagnostic.variable(42)   # Any variable by ID → int | None\nc.diagnostic.parameter(14)  # Any parameter by ID → int | None\n\n# All temperature channels\nfor temp in c.temperatures:\n    print(f\"  Channel {temp.id}: {temp.actual}° (on screen: {temp.on_main_screen})\")\n\n# Iterate fans, fuels, etc.\nfor fan in c.fans:\n    print(f\"  Fan {fan.id}: speed {fan.speed}\")\n```\n\n### Error handling\n\nThe exception hierarchy allows catching at any granularity:\n\n```python\nfrom fumis import (\n    Fumis,\n    FumisAuthenticationError,\n    FumisConnectionError,\n    FumisConnectionTimeoutError,\n    FumisError,\n    FumisResponseError,\n    FumisStoveOfflineError,\n)\n\ntry:\n    async with Fumis(mac=\"AABBCCDDEEFF\", password=\"1234\") as fumis:\n        await fumis.turn_on()\nexcept FumisAuthenticationError:\n    # Invalid MAC address or PIN (HTTP 401)\n    ...\nexcept FumisStoveOfflineError:\n    # WiRCU not connected to cloud (HTTP 404)\n    ...\nexcept FumisConnectionTimeoutError:\n    # Request timed out\n    ...\nexcept FumisConnectionError:\n    # Any connectivity issue (timeout, DNS, HTTP error)\n    ...\nexcept FumisError:\n    # Any Fumis-specific error\n    ...\n```\n\nException hierarchy:\n\n```\nFumisError\n├── FumisConnectionError\n│   └── FumisConnectionTimeoutError\n├── FumisResponseError\n├── FumisAuthenticationError\n└── FumisStoveOfflineError\n```\n\n### Enums\n\nStatus codes use proper Python enums, booleans where they make sense:\n\n```python\nfrom fumis import StoveStatus\n\ninfo.controller.stove_status == StoveStatus.COMBUSTION  # True\ninfo.controller.on                                      # True\ninfo.controller.eco_mode.enabled                        # False\n\n# Unknown values from the API are handled gracefully\nStoveStatus(999)  # StoveStatus.UNKNOWN (never crashes)\n```\n\n## Changelog \u0026 Releases\n\nThis repository keeps a change log using [GitHub's releases][releases]\nfunctionality. The format of the log is based on\n[Keep a Changelog][keepchangelog].\n\nReleases are based on [Semantic Versioning][semver], and use the format\nof `MAJOR.MINOR.PATCH`. In a nutshell, the version will be incremented\nbased on the following:\n\n- `MAJOR`: Incompatible or major changes.\n- `MINOR`: Backwards-compatible new features and enhancements.\n- `PATCH`: Backwards-compatible bugfixes and package updates.\n\n## Contributing\n\nThis is an active open-source project. We are always open to people who want to\nuse the code or contribute to it.\n\nWe've set up a separate document for our\n[contribution guidelines](CONTRIBUTING.md).\n\nFor in-depth API documentation and reverse engineering notes, see\n[RESEARCH.md](RESEARCH.md).\n\nThank you for being involved! :heart_eyes:\n\n## Setting up development environment\n\nThe easiest way to start, is by opening a CodeSpace here on GitHub, or by using\nthe [Dev Container][devcontainer] feature of Visual Studio Code.\n\n[![Open in Dev Containers][devcontainer-shield]][devcontainer]\n\nThis Python project is fully managed using the [Poetry][poetry] dependency\nmanager. But also relies on the use of NodeJS for certain checks during\ndevelopment.\n\nYou need at least:\n\n- Python 3.11+\n- [Poetry][poetry-install]\n- NodeJS 24+ (including NPM)\n\nTo install all packages, including all development requirements:\n\n```bash\nnpm install\npoetry install\n```\n\nAs this repository uses the [prek][prek] framework, all changes\nare linted and tested with each commit. You can run all checks and tests\nmanually, using the following command:\n\n```bash\npoetry run prek run --all-files\n```\n\nTo run just the Python tests:\n\n```bash\npoetry run pytest\n```\n\n## Authors \u0026 contributors\n\nThe original setup of this repository is by [Franck Nijhof][frenck].\n\nFor a full list of all authors and contributors,\ncheck [the contributor's page][contributors].\n\n## Disclaimer\n\nThis project is an independent, community-driven effort. It is **not\naffiliated with, endorsed by, or supported by** Fumis (ATech elektronika\nd.o.o.) or any stove manufacturer. All product names, trademarks, and\nregistered trademarks are property of their respective owners.\n\nThis library was developed by observing network traffic from devices we own,\nusing publicly available service manuals, and building on existing community\nintegrations. No access controls were bypassed and no proprietary code was\nused. This work is done for interoperability purposes, protected under EU\nlaw including the Software Directive (2009/24/EC), the Data Act (2023/2854),\nand the GDPR. See [RESEARCH.md](RESEARCH.md) for the full legal basis and\nmethodology.\n\nUse this software at your own risk. The authors are not responsible for any\ndamage to your stove, property, or person resulting from the use of this\nlibrary. Pellet stoves involve fire and heat; always follow your\nmanufacturer's safety guidelines.\n\n## License\n\nMIT License\n\nCopyright (c) 2019-2026 Franck Nijhof\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n\n[build-shield]: https://github.com/frenck/python-fumis/actions/workflows/tests.yaml/badge.svg\n[build]: https://github.com/frenck/python-fumis/actions/workflows/tests.yaml\n[codecov-shield]: https://codecov.io/gh/frenck/python-fumis/branch/main/graph/badge.svg\n[codecov]: https://codecov.io/gh/frenck/python-fumis\n[contributors]: https://github.com/frenck/python-fumis/graphs/contributors\n[devcontainer-shield]: https://img.shields.io/static/v1?label=Dev%20Containers\u0026message=Open\u0026color=blue\u0026logo=visualstudiocode\n[devcontainer]: https://vscode.dev/redirect?url=vscode://ms-vscode-remote.remote-containers/cloneInVolume?url=https://github.com/frenck/python-fumis\n[frenck]: https://github.com/frenck\n[github-sponsors-shield]: https://frenck.dev/wp-content/uploads/2019/12/github_sponsor.png\n[github-sponsors]: https://github.com/sponsors/frenck\n[keepchangelog]: http://keepachangelog.com/en/1.0.0/\n[license-shield]: https://img.shields.io/github/license/frenck/python-fumis.svg\n[maintenance-shield]: https://img.shields.io/maintenance/yes/2026.svg\n[patreon-shield]: https://frenck.dev/wp-content/uploads/2019/12/patreon.png\n[patreon]: https://www.patreon.com/frenck\n[poetry-install]: https://python-poetry.org/docs/#installation\n[poetry]: https://python-poetry.org\n[prek]: https://github.com/frenck/prek\n[project-stage-shield]: https://img.shields.io/badge/project%20stage-experimental-yellow.svg\n[pypi]: https://pypi.org/project/fumis/\n[python-versions-shield]: https://img.shields.io/pypi/pyversions/fumis\n[releases-shield]: https://img.shields.io/github/release/frenck/python-fumis.svg\n[releases]: https://github.com/frenck/python-fumis/releases\n[scorecard]: https://scorecard.dev/viewer/?uri=github.com/frenck/python-fumis\n[scorecard-shield]: https://api.scorecard.dev/projects/github.com/frenck/python-fumis/badge\n[semver]: http://semver.org/spec/v2.0.0.html\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Ffrenck%2Fpython-fumis","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Ffrenck%2Fpython-fumis","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Ffrenck%2Fpython-fumis/lists"}