dirigera


Namedirigera JSON
Version 1.1.7 PyPI version JSON
download
home_pageNone
SummaryAn unofficial Python client for controlling the IKEA Dirigera Smart Home Hub
upload_time2024-05-13 07:30:44
maintainerNone
docs_urlNone
authorNone
requires_python>=3.7
licenseMIT License Copyright (c) 2023 Leggin Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
keywords python iot smarthome hub lighting ikea tradfri dirigera
VCS
bugtrack_url
requirements requests websocket-client pydantic
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Dirigera Python Client

![Test](https://github.com/Leggin/dirigera/actions/workflows/tests.yml/badge.svg)
![Pypi](https://img.shields.io/pypi/v/dirigera)
[![Downloads](https://static.pepy.tech/badge/dirigera/month)](https://pepy.tech/project/dirigera)
![Downloads](https://img.shields.io/pypi/pyversions/dirigera)

This repository provides an unofficial Python client for controlling the IKEA Dirigera Smart Home Hub. Current features:

- [light control](#controlling-lights)
- [outlet control](#controlling-outlets)
- [air purifier control](#controlling-air-purifier)
- [blinds control](#controlling-blinds)
- [remote controllers](#remote-controllers) (tested with STYRBAR)
- [environment sensor](#environment-sensor) (tested with VINDSTYRKA)
- [scene](#scene)
- [motion sensor](#motion-sensor)
- [open/close sensor](#open-close-sensor)
- [event listener](#event-listener) for hub events

Support for other features will be added in the future and your input in form of issues and PRs is greatly appreciated.

## Installation

```bash
pip install dirigera
```

## Quickstart

1. Find out the ip-address of your Dirigera (check your router)
2. Once you installed `dirigera` with pip you can run the included generate-token script. Here you can directly set the ip-address of you dirigera as parameter.
   ```bash
   generate-token <Dirigera ip-address>
   ```
3. The script starts the auth process. When prompted, you must push the action button on the bottom of your Dirigera.
4. After that hit ENTER and your `token` will be printed to the console.  
   Example:
   ```
   Press the action button on Dirigera then hit ENTER ...
   Your Token:
   mgwB.aXqwpzV89N0aUwBhZMJjD8a.UBPyzy2InGtqgwo2MO5.xX4ug7.uBcVJquwYzLnAijF7SdYKvNxTo0uzQKahV10A-3ZQOz-UAubGP6sHWt1CJx3QmWZyE7ZcMZKgODXjSzWL1lumKgGz5dUIwFi3rhNxgK-IsBGeGVhNXPt8vGrYEcZePwPvNAIg8RqmlH27L-JZPnkAtP2wHoOdW72Djot3yJsohtEsb0p9mJvoZFSavTlTr4LDuf584vuH5fha5xoR9QhhIvvgbAP-s4EHFqENNi6vrYLHKR.sdqnv4sYw6UH-l6oiPnnRLxinoqBPOlWhlcL9doFviXQE.tZ9X8WVqyBrd0NYHlo9iorEvUbnZuD02BEJrg4NLwgh3rZtyF0Mi46HenynzBohbPn4RnuSYYCiHt5EZnWedxBtDqc7mSTm1ZtyD
   ```
5. Done. Use this token in the hub setup.
   ```
   dirigera.Hub(
      token="mgwB.aXqwpzV89N0aUwBhZMJjD8a...",
      ip_address="192.1..."
   )
   ```

## [Dirigera Hub](./src/dirigera/hub/hub.py)

Setting up the client works by providing the token and ip address.

```python
import dirigera

dirigera_hub = dirigera.Hub(
    token="mgwB.aXqwpzV89N0aUwBhZMJjD8a...",
    ip_address="192.1..."
)
```

# [Devices](./src/dirigera/devices/device.py)

All available devices (Light, Controller, Outlet, ...) consist of the core data defined in [device.py](./src/dirigera/devices/device.py):

### Core Device Data

```python
id: str
relation_id: Optional[str] = None
type: str
device_type: str
created_at: datetime.datetime
is_reachable: bool
last_seen: datetime.datetime
attributes: Attributes
capabilities: Capabilities
room: Room
device_set: List
remote_links: List[str]
is_hidden: Optional[bool] = None
```

### Attributes

All devices have attributes. Some devices have special attributes (for example Light has `is_on``). These are the core attributes each device has:

```python
custom_name: str
model: str
manufacturer: str
firmware_version: str
hardware_version: str
serial_number: Optional[str] = None
product_code: Optional[str] = None
ota_status: Optional[str] = None
ota_state: Optional[str] = None
ota_progress: Optional[int] = None
ota_policy: Optional[str] = None
ota_schedule_start: Optional[datetime.time] = None
ota_schedule_end: Optional[datetime.time] = None
```

### Capabilities

All devices have capabilities (for some it is just empty lists). Capabilities desrcibe what send/receive actions can be performed:

```python
can_send: List[str]
can_receive: List[str]
```

All devices have a room with the corresponging infos.

### Room

```python
id: str
name: str
color: str
icon: str
```

## [Controlling Lights](./src/dirigera/devices/light.py)

To get information about the available lights, you can use the `get_lights()` method:

```python
lights = dirigera_hub.get_lights()
```

The light object has the following attributes (additional to the core attributes):

```python
startup_on_off: Optional[StartupEnum] = None # Optional attributes are not present on all lights
is_on: bool
light_level: Optional[int] = None
color_temperature: Optional[int] = None
color_temperature_min: Optional[int] = None
color_temperature_max: Optional[int] = None
color_hue: Optional[int] = None
color_saturation: Optional[float] = None
```

Available methods for light are:

```python
light.set_name(name="kitchen light 1")

light.set_light(lamp_on=True)

light.set_light_level(light_level=90)

light.set_color_temperature(color_temp=3000)

light.set_light_color(hue=128, saturation=0.5)

light.set_startup_behaviour(behaviour=StartupEnum.START_OFF)
```

## [Controlling Outlets](./src/dirigera/devices/outlet.py)

To get information about the available outlets, you can use the `get_outlets()` method:

```python
outlets = dirigera_hub.get_outlets()
```

The outlet object has the following attributes (additional to the core attributes):

```python
is_on: bool
startup_on_off: Optional[StartupEnum] = None
```

Available methods for outlet are:

```python
outlet.set_name(name="kitchen socket 1")

outlet.set_on(outlet_on=True)

outlet.set_startup_behaviour(behaviour=StartupEnum.START_OFF)
```

## [Controlling Air Purifier](./src/dirigera/devices/air_purifier.py)

To get information about the available air purifiers, you can use the `get_air_purifiers()` method:

```python
air_purifiers = dirigera_hub.get_air_purifiers()
```

The air purifier object has the following attributes (additional to the core attributes):

```python
fan_mode: FanModeEnum
fan_mode_sequence: str
motor_state: int
child_lock: bool
status_light: bool
motor_runtime: int
filter_alarm_status: bool
filter_elapsed_time: int
filter_lifetime: int
current_p_m25: int
```

Available methods for blinds are:

```python
air_purifier.set_name(name="living room purifier")

air_purifier.set_fan_mode(fan_mode=FanModeEnum.AUTO)

air_purifier.set_motor_state(motor_state=42)

air_purifier.set_child_lock(child_lock=True)

air_purifier.set_status_light(light_state=False)
```

## [Controlling Blinds](./src/dirigera/devices/blinds.py)

To get information about the available blinds, you can use the `get_blinds()` method:

```python
blinds = dirigera_hub.get_blinds()
```

The blind object has the following attributes (additional to the core attributes):

```python
blinds_current_level: Optional[int] = None
blinds_target_level: Optional[int] = None
blinds_state: Optional[str] = None
```

Available methods for blinds are:

```python
blind.set_name(name="kitchen blind 1")

blind.set_target_level(target_level=90)
```

## [Remote Controllers](./src/dirigera/devices/controller.py)

Currently only tested with the STYRBAR remote.

To get information about the available controllers, you can use the `get_controllers()` method:

```python
controllers = dirigera_hub.get_controllers()
```

The controller object has the following attributes (additional to the core attributes):

```python
is_on: bool
battery_percentage: Optional[int] = None
switch_label: Optional[str] = None
```

Available methods for controller are:

```python
controller.set_name(name="kitchen remote 1")
```

## [Environment Sensor](./src/dirigera/devices/environment_sensor.py)

Currently only tested with the VINDSTYRKA sensor. If you have other sensors please send me the json and I will add support or create a PR.

To get the environment sensors use:

```python
sensors = dirigera_hub.get_environment_sensors()
```

The environment sensor object has the following attributes (additional to the core attributes):

```python
current_temperature: float
current_r_h: int # current humidity
current_p_m25: int # current particulate matter 2.5
max_measured_p_m25: int # maximum measurable particulate matter 2.5
min_measured_p_m25: int # minimum measurable particulate matter 2.5
voc_index: int # current volatile organic compound
```

Available methods for environment sensor are:

```python
sensor.set_name(name="Bathroom Sensor")
```

# [Scene](./src/dirigera/devices/scene.py)

To get the scenes use:

```python
scenes = dirigera_hub.get_scenes()
```

The scene object has the following attributes:

```python
id: str
type: SceneType
info: Info
triggers: List[Trigger]
actions: List[Action]
created_at: datetime.datetime
last_completed: Optional[datetime.datetime] = None
last_triggered: Optional[datetime.datetime] = None
last_undo: Optional[datetime.datetime] = None
commands: List[str]
undo_allowed_duration: int
```

Details to the Trigger, Action and Info class can be found in [scene.py](./src/dirigera/devices/scene.py)

Available methods for scene are:

```python
scene.trigger()
scene.undo()
```

### Creating a Scene

To create a scene use the `create_scene()` function.  
Example how to create an empty scene:

```python
scene = dirigera_hub.create_scene(
    info=Info(name="This is empty", icon=Icon.SCENES_BOOK),
)
```

Actions look like this:

```python
class Action(BaseIkeaModel):
    id: str
    type: str
    enabled: Optional[bool] = None
    attributes: Optional[ActionAttributes] = None
```

Example how create scene with action:

```python
from dirigera.devices.scene import Info, Icon, SceneType, Action, ActionAttributes

light = dirigera_hub.get_light_by_name("kitchen_lamp")

scene = dirigera_hub.create_scene(
    info=Info(name="Scene with action", icon=Icon.SCENES_BOOK),
    scene_type=SceneType.USER_SCENE,
    triggers=[],
    actions=[Action(id=light.id, type="device", enabled=True, attributes=ActionAttributes(is_on=False))],
)
```

Triggers look like this:

```python
class Trigger(BaseIkeaModel):
    id: Optional[str] = None
    type: str
    triggered_at: Optional[datetime.datetime] = None
    disabled: bool
    trigger: Optional[TriggerDetails] = None
```

Example how to create scene with trigger:
```python
from dirigera.devices.scene import Info, Icon, Trigger, SceneType, TriggerDetails, ControllerType, ClickPattern

scene = dirigera_hub.create_scene(
   info=Info(name="Scene with trigger", icon=Icon.SCENES_HEART),
   scene_type=SceneType.USER_SCENE,
   triggers=[
       Trigger(type="app", disabled=False),
       Trigger(type="controller", disabled=False,
               trigger=TriggerDetails(clickPattern=ClickPattern.SINGLE_PRESS, buttonIndex=0,
                                      deviceId="0000aaaa-0000-0000-aa00-0a0aa0a000a0_1",
                                      controllerType=ControllerType.SHORTCUT_CONTROLLER))])
```

All available icons can be found here: [Icons](./src/dirigera/devices/scene.py)

## [Motion Sensor](./src/dirigera/devices/motion_sensor.py)

To get information about the available motion sensors, you can use the `get_motion_sensors()` method:

```python
motions_sensors = dirigera_hub.get_motion_sensors()
```

The motion sensor object has the following attributes (additional to the core attributes):

```python
battery_percentage: int
is_on: bool
light_level: Optional[float]
```

Available methods for outlet are:

```python
motions_sensor.set_name(name="kitchen sensor 1")
```

## [Open Close Sensor](./src/dirigera/devices/open_close_sensor.py)

To get information about the available open/close sensors, you can use the `get_open_close_sensors()` method:

```python
open_close_sensors = dirigera_hub.get_open_close_sensors()
```

The open_close_sensor object has the following attributes (additional to the core attributes):

```python
is_open: bool
```

Available methods for outlet are:

```python
open_close_sensor.set_name(name="window 1")
```

## Event Listener

The event listener allows you to listen to events that are published by your Dirigera hub. This is useful if you want to automate tasks based on events such as when a light is turned on or off, or when the color temperature of a light is changed.

```python
import json
from typing import Any


def on_message(ws: Any, message: str):
    message_dict = json.loads(message)
    data = message_dict["data"]
    if data["id"] == bed_light.light_id:
        print(f"{message_dict['type']} event on {bed_light.custom_name}, attributes: {data['attributes']}")

def on_error(ws: Any, message: str):
    print(message)

dirigera_hub.create_event_listener(
    on_message=on_message, on_error=on_error
)
```

```
deviceStateChanged event on Bed Light, attributes: {'isOn': False}
```

## Motivation

The primary motivation for this project was to provide users with the ability to control the startup behavior of their smart home lamps when there is a power outage.  
The default behavior of the hub is to turn on all lights when power is restored, which can be problematic if the user is away from home or on vacation, and a small power fluctuation causes all lights to turn on and stay on. Unfortunately, the IKEA app does not offer a way to change this default behavior.  
The `set_startup_behaviour()` function enables users to override the default behavior and choose the startup behavior that best suits their needs (START_ON = turn on light when power is back, START_OFF = light stays off when power is back).  
I can not guarantee that all IKEA lamps offer this functionality.
EDIT: This is now an exposed feature in the app.

## Contributing

Contributions are welcome! If you have an idea for a new feature or a bug fix, please post and issue or submit a pull request.

### Setup of dev

For setting up the dev environment I recommend running the `setup.sh` script, which will create a venv and install the `requirements.txt` as well as the `dev-requirements.txt`.

### Tests

To run the tests execute the `run-test.sh` script or just run `pytest .`  
For linting you can run the `run-pylint.sh`.  
For types you can run the `run-mypy.sh`  
To test the different python versions you can use the `run-python-verions-test.sh` (this requires a running docker installation).  
All of these tests are also run when a PR is openend (and the test run is triggered).

## License

The MIT License (MIT)

Copyright (c) 2023 Leggin

            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "dirigera",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": null,
    "keywords": "python, iot, smarthome, hub, lighting, ikea, tradfri, dirigera",
    "author": null,
    "author_email": "Leggin <legginsun@gmail.com>",
    "download_url": "https://files.pythonhosted.org/packages/f1/dd/d9493cdc8902cd7655c4262fb9ca28e25fd7317925225e81fb8b8722c557/dirigera-1.1.7.tar.gz",
    "platform": null,
    "description": "# Dirigera Python Client\n\n![Test](https://github.com/Leggin/dirigera/actions/workflows/tests.yml/badge.svg)\n![Pypi](https://img.shields.io/pypi/v/dirigera)\n[![Downloads](https://static.pepy.tech/badge/dirigera/month)](https://pepy.tech/project/dirigera)\n![Downloads](https://img.shields.io/pypi/pyversions/dirigera)\n\nThis repository provides an unofficial Python client for controlling the IKEA Dirigera Smart Home Hub. Current features:\n\n- [light control](#controlling-lights)\n- [outlet control](#controlling-outlets)\n- [air purifier control](#controlling-air-purifier)\n- [blinds control](#controlling-blinds)\n- [remote controllers](#remote-controllers) (tested with STYRBAR)\n- [environment sensor](#environment-sensor) (tested with VINDSTYRKA)\n- [scene](#scene)\n- [motion sensor](#motion-sensor)\n- [open/close sensor](#open-close-sensor)\n- [event listener](#event-listener) for hub events\n\nSupport for other features will be added in the future and your input in form of issues and PRs is greatly appreciated.\n\n## Installation\n\n```bash\npip install dirigera\n```\n\n## Quickstart\n\n1. Find out the ip-address of your Dirigera (check your router)\n2. Once you installed `dirigera` with pip you can run the included generate-token script. Here you can directly set the ip-address of you dirigera as parameter.\n   ```bash\n   generate-token <Dirigera ip-address>\n   ```\n3. The script starts the auth process. When prompted, you must push the action button on the bottom of your Dirigera.\n4. After that hit ENTER and your `token` will be printed to the console.  \n   Example:\n   ```\n   Press the action button on Dirigera then hit ENTER ...\n   Your Token:\n   mgwB.aXqwpzV89N0aUwBhZMJjD8a.UBPyzy2InGtqgwo2MO5.xX4ug7.uBcVJquwYzLnAijF7SdYKvNxTo0uzQKahV10A-3ZQOz-UAubGP6sHWt1CJx3QmWZyE7ZcMZKgODXjSzWL1lumKgGz5dUIwFi3rhNxgK-IsBGeGVhNXPt8vGrYEcZePwPvNAIg8RqmlH27L-JZPnkAtP2wHoOdW72Djot3yJsohtEsb0p9mJvoZFSavTlTr4LDuf584vuH5fha5xoR9QhhIvvgbAP-s4EHFqENNi6vrYLHKR.sdqnv4sYw6UH-l6oiPnnRLxinoqBPOlWhlcL9doFviXQE.tZ9X8WVqyBrd0NYHlo9iorEvUbnZuD02BEJrg4NLwgh3rZtyF0Mi46HenynzBohbPn4RnuSYYCiHt5EZnWedxBtDqc7mSTm1ZtyD\n   ```\n5. Done. Use this token in the hub setup.\n   ```\n   dirigera.Hub(\n      token=\"mgwB.aXqwpzV89N0aUwBhZMJjD8a...\",\n      ip_address=\"192.1...\"\n   )\n   ```\n\n## [Dirigera Hub](./src/dirigera/hub/hub.py)\n\nSetting up the client works by providing the token and ip address.\n\n```python\nimport dirigera\n\ndirigera_hub = dirigera.Hub(\n    token=\"mgwB.aXqwpzV89N0aUwBhZMJjD8a...\",\n    ip_address=\"192.1...\"\n)\n```\n\n# [Devices](./src/dirigera/devices/device.py)\n\nAll available devices (Light, Controller, Outlet, ...) consist of the core data defined in [device.py](./src/dirigera/devices/device.py):\n\n### Core Device Data\n\n```python\nid: str\nrelation_id: Optional[str] = None\ntype: str\ndevice_type: str\ncreated_at: datetime.datetime\nis_reachable: bool\nlast_seen: datetime.datetime\nattributes: Attributes\ncapabilities: Capabilities\nroom: Room\ndevice_set: List\nremote_links: List[str]\nis_hidden: Optional[bool] = None\n```\n\n### Attributes\n\nAll devices have attributes. Some devices have special attributes (for example Light has `is_on``). These are the core attributes each device has:\n\n```python\ncustom_name: str\nmodel: str\nmanufacturer: str\nfirmware_version: str\nhardware_version: str\nserial_number: Optional[str] = None\nproduct_code: Optional[str] = None\nota_status: Optional[str] = None\nota_state: Optional[str] = None\nota_progress: Optional[int] = None\nota_policy: Optional[str] = None\nota_schedule_start: Optional[datetime.time] = None\nota_schedule_end: Optional[datetime.time] = None\n```\n\n### Capabilities\n\nAll devices have capabilities (for some it is just empty lists). Capabilities desrcibe what send/receive actions can be performed:\n\n```python\ncan_send: List[str]\ncan_receive: List[str]\n```\n\nAll devices have a room with the corresponging infos.\n\n### Room\n\n```python\nid: str\nname: str\ncolor: str\nicon: str\n```\n\n## [Controlling Lights](./src/dirigera/devices/light.py)\n\nTo get information about the available lights, you can use the `get_lights()` method:\n\n```python\nlights = dirigera_hub.get_lights()\n```\n\nThe light object has the following attributes (additional to the core attributes):\n\n```python\nstartup_on_off: Optional[StartupEnum] = None # Optional attributes are not present on all lights\nis_on: bool\nlight_level: Optional[int] = None\ncolor_temperature: Optional[int] = None\ncolor_temperature_min: Optional[int] = None\ncolor_temperature_max: Optional[int] = None\ncolor_hue: Optional[int] = None\ncolor_saturation: Optional[float] = None\n```\n\nAvailable methods for light are:\n\n```python\nlight.set_name(name=\"kitchen light 1\")\n\nlight.set_light(lamp_on=True)\n\nlight.set_light_level(light_level=90)\n\nlight.set_color_temperature(color_temp=3000)\n\nlight.set_light_color(hue=128, saturation=0.5)\n\nlight.set_startup_behaviour(behaviour=StartupEnum.START_OFF)\n```\n\n## [Controlling Outlets](./src/dirigera/devices/outlet.py)\n\nTo get information about the available outlets, you can use the `get_outlets()` method:\n\n```python\noutlets = dirigera_hub.get_outlets()\n```\n\nThe outlet object has the following attributes (additional to the core attributes):\n\n```python\nis_on: bool\nstartup_on_off: Optional[StartupEnum] = None\n```\n\nAvailable methods for outlet are:\n\n```python\noutlet.set_name(name=\"kitchen socket 1\")\n\noutlet.set_on(outlet_on=True)\n\noutlet.set_startup_behaviour(behaviour=StartupEnum.START_OFF)\n```\n\n## [Controlling Air Purifier](./src/dirigera/devices/air_purifier.py)\n\nTo get information about the available air purifiers, you can use the `get_air_purifiers()` method:\n\n```python\nair_purifiers = dirigera_hub.get_air_purifiers()\n```\n\nThe air purifier object has the following attributes (additional to the core attributes):\n\n```python\nfan_mode: FanModeEnum\nfan_mode_sequence: str\nmotor_state: int\nchild_lock: bool\nstatus_light: bool\nmotor_runtime: int\nfilter_alarm_status: bool\nfilter_elapsed_time: int\nfilter_lifetime: int\ncurrent_p_m25: int\n```\n\nAvailable methods for blinds are:\n\n```python\nair_purifier.set_name(name=\"living room purifier\")\n\nair_purifier.set_fan_mode(fan_mode=FanModeEnum.AUTO)\n\nair_purifier.set_motor_state(motor_state=42)\n\nair_purifier.set_child_lock(child_lock=True)\n\nair_purifier.set_status_light(light_state=False)\n```\n\n## [Controlling Blinds](./src/dirigera/devices/blinds.py)\n\nTo get information about the available blinds, you can use the `get_blinds()` method:\n\n```python\nblinds = dirigera_hub.get_blinds()\n```\n\nThe blind object has the following attributes (additional to the core attributes):\n\n```python\nblinds_current_level: Optional[int] = None\nblinds_target_level: Optional[int] = None\nblinds_state: Optional[str] = None\n```\n\nAvailable methods for blinds are:\n\n```python\nblind.set_name(name=\"kitchen blind 1\")\n\nblind.set_target_level(target_level=90)\n```\n\n## [Remote Controllers](./src/dirigera/devices/controller.py)\n\nCurrently only tested with the STYRBAR remote.\n\nTo get information about the available controllers, you can use the `get_controllers()` method:\n\n```python\ncontrollers = dirigera_hub.get_controllers()\n```\n\nThe controller object has the following attributes (additional to the core attributes):\n\n```python\nis_on: bool\nbattery_percentage: Optional[int] = None\nswitch_label: Optional[str] = None\n```\n\nAvailable methods for controller are:\n\n```python\ncontroller.set_name(name=\"kitchen remote 1\")\n```\n\n## [Environment Sensor](./src/dirigera/devices/environment_sensor.py)\n\nCurrently only tested with the VINDSTYRKA sensor. If you have other sensors please send me the json and I will add support or create a PR.\n\nTo get the environment sensors use:\n\n```python\nsensors = dirigera_hub.get_environment_sensors()\n```\n\nThe environment sensor object has the following attributes (additional to the core attributes):\n\n```python\ncurrent_temperature: float\ncurrent_r_h: int # current humidity\ncurrent_p_m25: int # current particulate matter 2.5\nmax_measured_p_m25: int # maximum measurable particulate matter 2.5\nmin_measured_p_m25: int # minimum measurable particulate matter 2.5\nvoc_index: int # current volatile organic compound\n```\n\nAvailable methods for environment sensor are:\n\n```python\nsensor.set_name(name=\"Bathroom Sensor\")\n```\n\n# [Scene](./src/dirigera/devices/scene.py)\n\nTo get the scenes use:\n\n```python\nscenes = dirigera_hub.get_scenes()\n```\n\nThe scene object has the following attributes:\n\n```python\nid: str\ntype: SceneType\ninfo: Info\ntriggers: List[Trigger]\nactions: List[Action]\ncreated_at: datetime.datetime\nlast_completed: Optional[datetime.datetime] = None\nlast_triggered: Optional[datetime.datetime] = None\nlast_undo: Optional[datetime.datetime] = None\ncommands: List[str]\nundo_allowed_duration: int\n```\n\nDetails to the Trigger, Action and Info class can be found in [scene.py](./src/dirigera/devices/scene.py)\n\nAvailable methods for scene are:\n\n```python\nscene.trigger()\nscene.undo()\n```\n\n### Creating a Scene\n\nTo create a scene use the `create_scene()` function.  \nExample how to create an empty scene:\n\n```python\nscene = dirigera_hub.create_scene(\n    info=Info(name=\"This is empty\", icon=Icon.SCENES_BOOK),\n)\n```\n\nActions look like this:\n\n```python\nclass Action(BaseIkeaModel):\n    id: str\n    type: str\n    enabled: Optional[bool] = None\n    attributes: Optional[ActionAttributes] = None\n```\n\nExample how create scene with action:\n\n```python\nfrom dirigera.devices.scene import Info, Icon, SceneType, Action, ActionAttributes\n\nlight = dirigera_hub.get_light_by_name(\"kitchen_lamp\")\n\nscene = dirigera_hub.create_scene(\n    info=Info(name=\"Scene with action\", icon=Icon.SCENES_BOOK),\n    scene_type=SceneType.USER_SCENE,\n    triggers=[],\n    actions=[Action(id=light.id, type=\"device\", enabled=True, attributes=ActionAttributes(is_on=False))],\n)\n```\n\nTriggers look like this:\n\n```python\nclass Trigger(BaseIkeaModel):\n    id: Optional[str] = None\n    type: str\n    triggered_at: Optional[datetime.datetime] = None\n    disabled: bool\n    trigger: Optional[TriggerDetails] = None\n```\n\nExample how to create scene with trigger:\n```python\nfrom dirigera.devices.scene import Info, Icon, Trigger, SceneType, TriggerDetails, ControllerType, ClickPattern\n\nscene = dirigera_hub.create_scene(\n   info=Info(name=\"Scene with trigger\", icon=Icon.SCENES_HEART),\n   scene_type=SceneType.USER_SCENE,\n   triggers=[\n       Trigger(type=\"app\", disabled=False),\n       Trigger(type=\"controller\", disabled=False,\n               trigger=TriggerDetails(clickPattern=ClickPattern.SINGLE_PRESS, buttonIndex=0,\n                                      deviceId=\"0000aaaa-0000-0000-aa00-0a0aa0a000a0_1\",\n                                      controllerType=ControllerType.SHORTCUT_CONTROLLER))])\n```\n\nAll available icons can be found here: [Icons](./src/dirigera/devices/scene.py)\n\n## [Motion Sensor](./src/dirigera/devices/motion_sensor.py)\n\nTo get information about the available motion sensors, you can use the `get_motion_sensors()` method:\n\n```python\nmotions_sensors = dirigera_hub.get_motion_sensors()\n```\n\nThe motion sensor object has the following attributes (additional to the core attributes):\n\n```python\nbattery_percentage: int\nis_on: bool\nlight_level: Optional[float]\n```\n\nAvailable methods for outlet are:\n\n```python\nmotions_sensor.set_name(name=\"kitchen sensor 1\")\n```\n\n## [Open Close Sensor](./src/dirigera/devices/open_close_sensor.py)\n\nTo get information about the available open/close sensors, you can use the `get_open_close_sensors()` method:\n\n```python\nopen_close_sensors = dirigera_hub.get_open_close_sensors()\n```\n\nThe open_close_sensor object has the following attributes (additional to the core attributes):\n\n```python\nis_open: bool\n```\n\nAvailable methods for outlet are:\n\n```python\nopen_close_sensor.set_name(name=\"window 1\")\n```\n\n## Event Listener\n\nThe event listener allows you to listen to events that are published by your Dirigera hub. This is useful if you want to automate tasks based on events such as when a light is turned on or off, or when the color temperature of a light is changed.\n\n```python\nimport json\nfrom typing import Any\n\n\ndef on_message(ws: Any, message: str):\n    message_dict = json.loads(message)\n    data = message_dict[\"data\"]\n    if data[\"id\"] == bed_light.light_id:\n        print(f\"{message_dict['type']} event on {bed_light.custom_name}, attributes: {data['attributes']}\")\n\ndef on_error(ws: Any, message: str):\n    print(message)\n\ndirigera_hub.create_event_listener(\n    on_message=on_message, on_error=on_error\n)\n```\n\n```\ndeviceStateChanged event on Bed Light, attributes: {'isOn': False}\n```\n\n## Motivation\n\nThe primary motivation for this project was to provide users with the ability to control the startup behavior of their smart home lamps when there is a power outage.  \nThe default behavior of the hub is to turn on all lights when power is restored, which can be problematic if the user is away from home or on vacation, and a small power fluctuation causes all lights to turn on and stay on. Unfortunately, the IKEA app does not offer a way to change this default behavior.  \nThe `set_startup_behaviour()` function enables users to override the default behavior and choose the startup behavior that best suits their needs (START_ON = turn on light when power is back, START_OFF = light stays off when power is back).  \nI can not guarantee that all IKEA lamps offer this functionality.\nEDIT: This is now an exposed feature in the app.\n\n## Contributing\n\nContributions are welcome! If you have an idea for a new feature or a bug fix, please post and issue or submit a pull request.\n\n### Setup of dev\n\nFor setting up the dev environment I recommend running the `setup.sh` script, which will create a venv and install the `requirements.txt` as well as the `dev-requirements.txt`.\n\n### Tests\n\nTo run the tests execute the `run-test.sh` script or just run `pytest .`  \nFor linting you can run the `run-pylint.sh`.  \nFor types you can run the `run-mypy.sh`  \nTo test the different python versions you can use the `run-python-verions-test.sh` (this requires a running docker installation).  \nAll of these tests are also run when a PR is openend (and the test run is triggered).\n\n## License\n\nThe MIT License (MIT)\n\nCopyright (c) 2023 Leggin\n",
    "bugtrack_url": null,
    "license": "MIT License  Copyright (c) 2023 Leggin  Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the \"Software\"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:  The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.  THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ",
    "summary": "An unofficial Python client for controlling the IKEA Dirigera Smart Home Hub",
    "version": "1.1.7",
    "project_urls": {
        "Homepage": "https://github.com/Leggin/dirigera"
    },
    "split_keywords": [
        "python",
        " iot",
        " smarthome",
        " hub",
        " lighting",
        " ikea",
        " tradfri",
        " dirigera"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d648780470831ce52c3635014a2d8bfb04fface098660757f28f7652a6c7672e",
                "md5": "a09c486f3056bc4a6db825f1d251d6df",
                "sha256": "5e9fa6540f1b6fca4b73bdebfec112d293ff496699b4f35ac608f56abba63218"
            },
            "downloads": -1,
            "filename": "dirigera-1.1.7-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "a09c486f3056bc4a6db825f1d251d6df",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 23890,
            "upload_time": "2024-05-13T07:30:42",
            "upload_time_iso_8601": "2024-05-13T07:30:42.643462Z",
            "url": "https://files.pythonhosted.org/packages/d6/48/780470831ce52c3635014a2d8bfb04fface098660757f28f7652a6c7672e/dirigera-1.1.7-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "f1ddd9493cdc8902cd7655c4262fb9ca28e25fd7317925225e81fb8b8722c557",
                "md5": "3db259fa0b44d040768fc208987228e9",
                "sha256": "0a037eef24d533cb72f59a553d204e4e8358f5662a659950cf8e652662555a4b"
            },
            "downloads": -1,
            "filename": "dirigera-1.1.7.tar.gz",
            "has_sig": false,
            "md5_digest": "3db259fa0b44d040768fc208987228e9",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 29893,
            "upload_time": "2024-05-13T07:30:44",
            "upload_time_iso_8601": "2024-05-13T07:30:44.718749Z",
            "url": "https://files.pythonhosted.org/packages/f1/dd/d9493cdc8902cd7655c4262fb9ca28e25fd7317925225e81fb8b8722c557/dirigera-1.1.7.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-05-13 07:30:44",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "Leggin",
    "github_project": "dirigera",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "requests",
            "specs": [
                [
                    "==",
                    "2.*"
                ]
            ]
        },
        {
            "name": "websocket-client",
            "specs": [
                [
                    "==",
                    "1.5.1"
                ]
            ]
        },
        {
            "name": "pydantic",
            "specs": [
                [
                    "==",
                    "2.4.2"
                ]
            ]
        }
    ],
    "lcname": "dirigera"
}
        
Elapsed time: 0.38712s