qn-scale-ble


Nameqn-scale-ble JSON
Version 0.0.1 PyPI version JSON
download
home_pageNone
SummaryAn unofficial Python package for interacting with QN-Scale smart scales using BLE. Not affiliated with the manufacturer or any of their subsidiaries.
upload_time2025-01-21 02:45:55
maintainerNone
docs_urlNone
authorNone
requires_python>=3.10
licenseNone
keywords ble bluetooth fitness scale health iot qn-scale smart home smart scale weight
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # QN Scale BLE

This package is a fork of https://github.com/ronnnnnnnnnnnnn/etekcity_esf551_ble that has been modified to connect to the FITINDEX ES-26M Bluetooth smart scale, internally named "QN-Scale". The modifications were based on the reverse engineering work and code in the [OpenScale project](https://github.com/oliexdev/openScale).

## Installation

Install the package using pip:

```bash
pip install qn_scale_ble
```


## Quick Start

Here's a basic example of how to use the library:

```python
import asyncio
from qn_scale_ble import (
    IMPEDANCE_KEY,
    WEIGHT_KEY,
    QnScale,
    ScaleData,
    WeightUnit,
    BodyMetrics,
    Sex,
)

async def main():
    def notification_callback(data: ScaleData):
        print(f"Weight: {data.measurements[WEIGHT_KEY]} kg")
        print(f"Display Unit: {data.display_unit.name}")
        if IMPEDANCE_KEY in data.measurements:
            print(f"Impedance: {data.measurements[IMPEDANCE_KEY]} Ω")
            
            # Calculate body metrics
            # Note: Replace with your actual height, age and sex
            body_metrics = BodyMetrics(
                weight_kg=data.measurements[WEIGHT_KEY],
                height_m=1.75,  # Example height
                age=30,  # Example age
                sex=Sex.Male,  # Example sex
                impedance=data.measurements[IMPEDANCE_KEY]
            )
            print(f"Body Mass Index: {body_metrics.body_mass_index:.2f}")
            print(f"Body Fat Percentage: {body_metrics.body_fat_percentage:.1f}%")
            print(f"Fat-Free Weight: {body_metrics.fat_free_weight:.2f} kg")
            print(f"Subcutaneous Fat Percentage: {body_metrics.subcutaneous_fat_percentage:.1f}%")
            print(f"Visceral Fat Value: {body_metrics.visceral_fat_value}")
            print(f"Body Water Percentage: {body_metrics.body_water_percentage:.1f}%")
            print(f"Basal Metabolic Rate: {body_metrics.basal_metabolic_rate} calories")
            print(f"Skeletal Muscle Percentage: {body_metrics.skeletal_muscle_percentage:.1f}%")
            print(f"Muscle Mass: {body_metrics.muscle_mass:.2f} kg")
            print(f"Bone Mass: {body_metrics.bone_mass:.2f} kg")
            print(f"Protein Percentage: {body_metrics.protein_percentage:.1f}%")
            print(f"Metabolic Age: {body_metrics.metabolic_age} years")

    # Replace XX:XX:XX:XX:XX:XX with your scale's Bluetooth address
    scale = QnScale("XX:XX:XX:XX:XX:XX", notification_callback)
    scale.display_unit = WeightUnit.KG  # Set display unit to kilograms

    await scale.async_start()
    await asyncio.sleep(30)  # Wait for measurements
    await scale.async_stop()

asyncio.run(main())
```
For a real-life usage example of this library, check out the [Etekcity Fitness Scale BLE Integration for Home Assistant](https://github.com/ronnnnnnnnnnnnn/etekcity_fitness_scale_ble).


## API Reference

### `QnScale`

The main class for interacting with the scale.

#### Methods:

- `__init__(self, address: str, notification_callback: Callable[[ScaleData], None], display_unit: WeightUnit = None)`
- `async_start()`: Start scanning for and connecting to the scale.
- `async_stop()`: Stop the connection to the scale.

#### Properties:

- `display_unit`: Get or set the display unit (WeightUnit.KG, WeightUnit.LB or WeightUnit.ST). Returns None if the display unit is currently unknown (not set by the user and not yet received from the scale together with a stable weight measurement).
- `hw_version`: Get the hardware version of the scale (read-only).
- `sw_version`: Get the software version of the scale (read-only).

### `QnScaleWithBodyMetrics`

An extended version of QnScale that automatically calculates body metrics.

#### Methods:

- `__init__(self, address: str, notification_callback: Callable[[ScaleData], None], sex: Sex, birthdate: date, height_m: float, display_unit: WeightUnit = None)`
- `async_start()`: Start scanning for and connecting to the scale.
- `async_stop()`: Stop the connection to the scale.

#### Properties:

- `display_unit`: Get or set the display unit (WeightUnit.KG, WeightUnit.LB or WeightUnit.ST). Returns None if the display unit is currently unknown (not set by the user and not yet received from the scale together with a stable weight measurement).
- `hw_version`: Get the hardware version of the scale (read-only).
- `sw_version`: Get the software version of the scale (read-only).

### `WeightUnit`

An enum representing the possible display units:

- `WeightUnit.KG`: Kilograms
- `WeightUnit.LB`: Pounds
- `WeightUnit.ST`: Stones

### `ScaleData`

A dataclass containing scale measurement data:

- `name`: Scale name
- `address`: Scale Bluetooth address
- `hw_version`: Hardware version
- `sw_version`: Software version
- `display_unit`: Current display unit (concerns only the weight as displayed on the scale, the measurement itself is always provided by the API in kilograms)
- `measurements`: Dictionary of measurements (currently supports: weight in kilograms and impedance in ohms)

### `BodyMetrics`

A class for calculating various body composition metrics based on height, age, sex, and the weight and impedance as measured by the scale, similar to the metrics calculated and shown in the VeSync app. Note that currently "Athlete Mode" is not supported.

#### Methods:

- `__init__(self, weight_kg: float, height_m: float, age: int, sex: Sex, impedance: int)`

#### Properties:

- `body_mass_index`: Body Mass Index (BMI)
- `body_fat_percentage`: Estimated body fat percentage
- `fat_free_weight`: Weight of non-fat body mass in kg
- `subcutaneous_fat_percentage`: Estimated subcutaneous fat percentage
- `visceral_fat_value`: Estimated visceral fat level (unitless)
- `body_water_percentage`: Estimated body water percentage
- `basal_metabolic_rate`: Estimated basal metabolic rate in calories
- `skeletal_muscle_percentage`: Estimated skeletal muscle percentage
- `muscle_mass`: Estimated muscle mass in kg
- `bone_mass`: Estimated bone mass in kg
- `protein_percentage`: Estimated protein percentage
- `weight_score`: Calculated weight score (0-100)
- `fat_score`: Calculated fat score (0-100)
- `bmi_score`: Calculated BMI score (0-100)
- `health_score`: Overall health score based on other metrics (0-100)
- `metabolic_age`: Estimated metabolic age in years

### `Sex`

An enum representing biological sex for body composition calculations:

- `Sex.Male`
- `Sex.Female`


## Compatibility

- Tested on Mac (Apple Silicon) and Raspberry Pi 4
- Compatibility with Windows is unknown


## Troubleshooting

On Raspberry Pi 4 (and possibly other Linux machines using BlueZ), if you encounter a `org.bluez.Error.InProgress` error, try the following in `bluetoothctl`:

```
power off
power on
scan on
```
(See https://github.com/home-assistant/core/issues/76186#issuecomment-1204954485)


## Support the Project

If you find this unofficial project helpful, consider buying the upstream project author(s) a coffee!

etekcity_esf551_ble: [![Buy Me A Coffee](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/ronnnnnnn)

OpenScale: [Donations](https://github.com/oliexdev/openScale?tab=readme-ov-file#donations-heart)

## License

This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.


## Disclaimer

This is an independent project developed by the community. It is not endorsed by, directly affiliated with, maintained, authorized, or sponsored by the scale manufacturers or any of their affiliates or subsidiaries. All product and company names are the registered trademarks of their original owners. The use of any trade name or trademark is for identification and reference purposes only and does not imply any association with the trademark holder of their product brand.

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "qn-scale-ble",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": null,
    "keywords": "ble, bluetooth, fitness scale, health, iot, qn-scale, smart home, smart scale, weight",
    "author": null,
    "author_email": "BHSPitMonkey <bhs2007@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/14/a4/9adad3d05f50902a787118c4936951fbeb8a5c9edae54ac9f74436e1cfa7/qn_scale_ble-0.0.1.tar.gz",
    "platform": null,
    "description": "# QN Scale BLE\n\nThis package is a fork of https://github.com/ronnnnnnnnnnnnn/etekcity_esf551_ble that has been modified to connect to the FITINDEX ES-26M Bluetooth smart scale, internally named \"QN-Scale\". The modifications were based on the reverse engineering work and code in the [OpenScale project](https://github.com/oliexdev/openScale).\n\n## Installation\n\nInstall the package using pip:\n\n```bash\npip install qn_scale_ble\n```\n\n\n## Quick Start\n\nHere's a basic example of how to use the library:\n\n```python\nimport asyncio\nfrom qn_scale_ble import (\n    IMPEDANCE_KEY,\n    WEIGHT_KEY,\n    QnScale,\n    ScaleData,\n    WeightUnit,\n    BodyMetrics,\n    Sex,\n)\n\nasync def main():\n    def notification_callback(data: ScaleData):\n        print(f\"Weight: {data.measurements[WEIGHT_KEY]} kg\")\n        print(f\"Display Unit: {data.display_unit.name}\")\n        if IMPEDANCE_KEY in data.measurements:\n            print(f\"Impedance: {data.measurements[IMPEDANCE_KEY]} \u03a9\")\n            \n            # Calculate body metrics\n            # Note: Replace with your actual height, age and sex\n            body_metrics = BodyMetrics(\n                weight_kg=data.measurements[WEIGHT_KEY],\n                height_m=1.75,  # Example height\n                age=30,  # Example age\n                sex=Sex.Male,  # Example sex\n                impedance=data.measurements[IMPEDANCE_KEY]\n            )\n            print(f\"Body Mass Index: {body_metrics.body_mass_index:.2f}\")\n            print(f\"Body Fat Percentage: {body_metrics.body_fat_percentage:.1f}%\")\n            print(f\"Fat-Free Weight: {body_metrics.fat_free_weight:.2f} kg\")\n            print(f\"Subcutaneous Fat Percentage: {body_metrics.subcutaneous_fat_percentage:.1f}%\")\n            print(f\"Visceral Fat Value: {body_metrics.visceral_fat_value}\")\n            print(f\"Body Water Percentage: {body_metrics.body_water_percentage:.1f}%\")\n            print(f\"Basal Metabolic Rate: {body_metrics.basal_metabolic_rate} calories\")\n            print(f\"Skeletal Muscle Percentage: {body_metrics.skeletal_muscle_percentage:.1f}%\")\n            print(f\"Muscle Mass: {body_metrics.muscle_mass:.2f} kg\")\n            print(f\"Bone Mass: {body_metrics.bone_mass:.2f} kg\")\n            print(f\"Protein Percentage: {body_metrics.protein_percentage:.1f}%\")\n            print(f\"Metabolic Age: {body_metrics.metabolic_age} years\")\n\n    # Replace XX:XX:XX:XX:XX:XX with your scale's Bluetooth address\n    scale = QnScale(\"XX:XX:XX:XX:XX:XX\", notification_callback)\n    scale.display_unit = WeightUnit.KG  # Set display unit to kilograms\n\n    await scale.async_start()\n    await asyncio.sleep(30)  # Wait for measurements\n    await scale.async_stop()\n\nasyncio.run(main())\n```\nFor a real-life usage example of this library, check out the [Etekcity Fitness Scale BLE Integration for Home Assistant](https://github.com/ronnnnnnnnnnnnn/etekcity_fitness_scale_ble).\n\n\n## API Reference\n\n### `QnScale`\n\nThe main class for interacting with the scale.\n\n#### Methods:\n\n- `__init__(self, address: str, notification_callback: Callable[[ScaleData], None], display_unit: WeightUnit = None)`\n- `async_start()`: Start scanning for and connecting to the scale.\n- `async_stop()`: Stop the connection to the scale.\n\n#### Properties:\n\n- `display_unit`: Get or set the display unit (WeightUnit.KG, WeightUnit.LB or WeightUnit.ST). Returns None if the display unit is currently unknown (not set by the user and not yet received from the scale together with a stable weight measurement).\n- `hw_version`: Get the hardware version of the scale (read-only).\n- `sw_version`: Get the software version of the scale (read-only).\n\n### `QnScaleWithBodyMetrics`\n\nAn extended version of QnScale that automatically calculates body metrics.\n\n#### Methods:\n\n- `__init__(self, address: str, notification_callback: Callable[[ScaleData], None], sex: Sex, birthdate: date, height_m: float, display_unit: WeightUnit = None)`\n- `async_start()`: Start scanning for and connecting to the scale.\n- `async_stop()`: Stop the connection to the scale.\n\n#### Properties:\n\n- `display_unit`: Get or set the display unit (WeightUnit.KG, WeightUnit.LB or WeightUnit.ST). Returns None if the display unit is currently unknown (not set by the user and not yet received from the scale together with a stable weight measurement).\n- `hw_version`: Get the hardware version of the scale (read-only).\n- `sw_version`: Get the software version of the scale (read-only).\n\n### `WeightUnit`\n\nAn enum representing the possible display units:\n\n- `WeightUnit.KG`: Kilograms\n- `WeightUnit.LB`: Pounds\n- `WeightUnit.ST`: Stones\n\n### `ScaleData`\n\nA dataclass containing scale measurement data:\n\n- `name`: Scale name\n- `address`: Scale Bluetooth address\n- `hw_version`: Hardware version\n- `sw_version`: Software version\n- `display_unit`: Current display unit (concerns only the weight as displayed on the scale, the measurement itself is always provided by the API in kilograms)\n- `measurements`: Dictionary of measurements (currently supports: weight in kilograms and impedance in ohms)\n\n### `BodyMetrics`\n\nA class for calculating various body composition metrics based on height, age, sex, and the weight and impedance as measured by the scale, similar to the metrics calculated and shown in the VeSync app. Note that currently \"Athlete Mode\" is not supported.\n\n#### Methods:\n\n- `__init__(self, weight_kg: float, height_m: float, age: int, sex: Sex, impedance: int)`\n\n#### Properties:\n\n- `body_mass_index`: Body Mass Index (BMI)\n- `body_fat_percentage`: Estimated body fat percentage\n- `fat_free_weight`: Weight of non-fat body mass in kg\n- `subcutaneous_fat_percentage`: Estimated subcutaneous fat percentage\n- `visceral_fat_value`: Estimated visceral fat level (unitless)\n- `body_water_percentage`: Estimated body water percentage\n- `basal_metabolic_rate`: Estimated basal metabolic rate in calories\n- `skeletal_muscle_percentage`: Estimated skeletal muscle percentage\n- `muscle_mass`: Estimated muscle mass in kg\n- `bone_mass`: Estimated bone mass in kg\n- `protein_percentage`: Estimated protein percentage\n- `weight_score`: Calculated weight score (0-100)\n- `fat_score`: Calculated fat score (0-100)\n- `bmi_score`: Calculated BMI score (0-100)\n- `health_score`: Overall health score based on other metrics (0-100)\n- `metabolic_age`: Estimated metabolic age in years\n\n### `Sex`\n\nAn enum representing biological sex for body composition calculations:\n\n- `Sex.Male`\n- `Sex.Female`\n\n\n## Compatibility\n\n- Tested on Mac (Apple Silicon) and Raspberry Pi 4\n- Compatibility with Windows is unknown\n\n\n## Troubleshooting\n\nOn Raspberry Pi 4 (and possibly other Linux machines using BlueZ), if you encounter a `org.bluez.Error.InProgress` error, try the following in `bluetoothctl`:\n\n```\npower off\npower on\nscan on\n```\n(See https://github.com/home-assistant/core/issues/76186#issuecomment-1204954485)\n\n\n## Support the Project\n\nIf you find this unofficial project helpful, consider buying the upstream project author(s) a coffee!\n\netekcity_esf551_ble: [![Buy Me A Coffee](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/ronnnnnnn)\n\nOpenScale: [Donations](https://github.com/oliexdev/openScale?tab=readme-ov-file#donations-heart)\n\n## License\n\nThis project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.\n\n\n## Disclaimer\n\nThis is an independent project developed by the community. It is not endorsed by, directly affiliated with, maintained, authorized, or sponsored by the scale manufacturers or any of their affiliates or subsidiaries. All product and company names are the registered trademarks of their original owners. The use of any trade name or trademark is for identification and reference purposes only and does not imply any association with the trademark holder of their product brand.\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "An unofficial Python package for interacting with QN-Scale smart scales using BLE. Not affiliated with the manufacturer or any of their subsidiaries.",
    "version": "0.0.1",
    "project_urls": {
        "Bug Tracker": "https://github.com/BHSPitMonkey/qn_scale_ble/issues",
        "Documentation": "https://github.com/BHSPitMonkey/qn_scale_ble#readme",
        "Homepage": "https://github.com/BHSPitMonkey/qn_scale_ble",
        "Source Code": "https://github.com/BHSPitMonkey/qn_scale_ble/"
    },
    "split_keywords": [
        "ble",
        " bluetooth",
        " fitness scale",
        " health",
        " iot",
        " qn-scale",
        " smart home",
        " smart scale",
        " weight"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "13d8f9630c42109dd216c3d05abda9d70a65dce0774facd6bde40b7de0093c71",
                "md5": "d70c71f7638d6055a13f05b7e4069556",
                "sha256": "0ff7b8749eab1d6f0363f9545d77fabfecee503c867a6829aa633ccac75587c5"
            },
            "downloads": -1,
            "filename": "qn_scale_ble-0.0.1-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "d70c71f7638d6055a13f05b7e4069556",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.10",
            "size": 13676,
            "upload_time": "2025-01-21T02:45:53",
            "upload_time_iso_8601": "2025-01-21T02:45:53.082189Z",
            "url": "https://files.pythonhosted.org/packages/13/d8/f9630c42109dd216c3d05abda9d70a65dce0774facd6bde40b7de0093c71/qn_scale_ble-0.0.1-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "14a49adad3d05f50902a787118c4936951fbeb8a5c9edae54ac9f74436e1cfa7",
                "md5": "deef48f8efe96127b714c93b91c2c661",
                "sha256": "2e95d4687ca933a6273e36d9c346ad0831740b1dbfd81ea3028a1d393ff08dcc"
            },
            "downloads": -1,
            "filename": "qn_scale_ble-0.0.1.tar.gz",
            "has_sig": false,
            "md5_digest": "deef48f8efe96127b714c93b91c2c661",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 12964,
            "upload_time": "2025-01-21T02:45:55",
            "upload_time_iso_8601": "2025-01-21T02:45:55.041415Z",
            "url": "https://files.pythonhosted.org/packages/14/a4/9adad3d05f50902a787118c4936951fbeb8a5c9edae54ac9f74436e1cfa7/qn_scale_ble-0.0.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-01-21 02:45:55",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "BHSPitMonkey",
    "github_project": "qn_scale_ble",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "qn-scale-ble"
}
        
Elapsed time: 1.36083s