Metadata-Version: 1.1
Name: appdaemontestframework
Version: 2.0.0
Summary: Clean, human-readable tests for Appdaemon
Home-page: https://floriankempenich.github.io/Appdaemon-Test-Framework
Author: Florian Kempenich
Author-email: Flori@nKempenich.com
License: MIT
Description: # Appdaemon Test Framework
        [![Travis](https://img.shields.io/travis/FlorianKempenich/Appdaemon-Test-Framework.svg)](https://travis-ci.org/FlorianKempenich/Appdaemon-Test-Framework) [![PyPI](https://img.shields.io/pypi/v/appdaemontestframework.svg)](https://pypi.org/project/appdaemontestframework/)
        
        Clean, human-readable tests for your Appdaemon automations.
        
        * Totally transparent, No code modification is needed.
        * Mock the state of your home: `given_that.state_of('sensor.temperature').is_set_to('24.9')`
        * Seamless assertions: `assert_that('light.bathroom').was.turned_on()`
        * Simulate time: `time_travel.fast_forward(2).minutes()`
        
        ##### How does it look?
        ```python
        def test_during_night_light_turn_on(given_that, living_room, assert_that):
            given_that.state_of('sensor.living_room_illumination').is_set_to(200) # 200lm == night
            living_room._new_motion(None, None, None)
            assert_that('light.living_room').was.turned_on()
        
        def test_click_light_turn_on_for_5_minutes(given_that, living_room, assert_that):
            living_room._new_button_click(None, None, None)
            assert_that('light.bathroom').was.turned_on()
        
            # At T=4min light should not yet have been turned off
            time_travel.fast_forward(4).minutes()    
            assert_that('light.bathroom').was_not.turned_off()
        
            # At T=5min light should have been turned off
            time_travel.fast_forward(1).minutes()    
            time_travel.assert_current_time(5).minutes()
            assert_that('light.bathroom').was.turned_off()
            
        ```
        
        ---
        ## Table of Contents
        
        - [5-Minutes Quick Start Guide](#5-minutes-quick-start-guide)
          * [Initial Setup](#initial-setup)
          * [Write you first unit test](#write-you-first-unit-test)
          * [Result](#result)
        - [General Test Flow and Available helpers](#general-test-flow-and-available-helpers)
          * [0. Initialize the automation: `@automation_fixture`](#0-initialize-the-automation-automation_fixture)
          * [1. Set the stage to prepare for the test: `given_that`](#1-set-the-stage-to-prepare-for-the-test-given_that)
          * [2. Trigger action on your automation](#2-trigger-action-on-your-automation)
          * [3. Assert on your way out: `assert_that`](#3-assert-on-your-way-out-assert_that)
          * [Bonus — Travel in Time: `time_travel`](#bonus--travel-in-time-time_travel)
        - [Examples](#examples)
        - [Under The Hood](#under-the-hood)
        - [Advanced Usage](#advanced-usage)
        - [Contributing](#contributing)
        - [Author Information](#author-information)
        
        ---
        
        ## 5-Minutes Quick Start Guide
        ### Initial Setup
        1. Install **pytest**: `pip install pytest`
        1. Install the **framework**: `pip install appdaemontestframework`
        1. Copy [**`conftest.py`**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/new-features/doc/full_example/conftest.py) at the **root** of your project
        
        ### Write you first unit test
        Let's test an Appdaemon automation we created, which, say, handles automatic lighting in the Living Room: `class LivingRoom`  
        <!-- We called the class `LivingRoom`. Since it's an Appdaemon automation, its lifecycle is handled  -->
        
        1. **Initialize** the Automation Under Test with the `@automation_fixture` decorator:
           ```python
           @automation_fixture(LivingRoom)
           def living_room():
             pass
           ```
        1. **Write your first test:**
           ##### Our first unit test
           ```python
           def test_during_night_light_turn_on(given_that, living_room, assert_that):
               given_that.state_of('sensor.living_room_illumination').is_set_to(200) # 200lm == night
               living_room._new_motion(None, None, None)
               assert_that('light.living_room').was.turned_on()      
           ```
           > ##### Note
           > The following fixtures are **injected** by pytest using the **[`conftest.py`](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/new-features/doc/full_example/conftest.py) file** and the **initialisation fixture created at Step 1**:
           > * `living_room`
           > * `given_that`
           > * `assert_that`
           > * `time_travel`
        
        ### Result
        ```python
        # Important:
        # For this example to work, do not forget to copy the `conftest.py` file.
        
        @automation_fixture(LivingRoom)
        def living_room():
            pass
        
        def test_during_night_light_turn_on(given_that, living_room, assert_that):
            given_that.state_of('sensor.living_room_illumination').is_set_to(200) # 200lm == night
            living_room._new_motion(None, None, None)
            assert_that('light.living_room').was.turned_on()
        
        def test_during_day_light_DOES_NOT_turn_on(given_that, living_room, assert_that):
            given_that.state_of('sensor.living_room_illumination').is_set_to(1000) # 1000lm == sunlight
            living_room._new_motion(None, None, None)
            assert_that('light.living_room').was_not.turned_on()
        ```
        
        
        ---
        ## General Test Flow and Available helpers
        ### 0. Initialize the automation: `@automation_fixture`
        ```python
        # Command
        @automation_fixture(AUTOMATION_CLASS)
        def FIXTURE_NAME():
            pass
          
        # Example
        @automation_fixture(LivingRoom)
        def living_room():
            pass
        ```
        The automation given to the fixture will be: 
        * **Created**  
          _Using the required mocks provided by the framework_
        * **Initialized**  
          _By calling the `initialize()` function_
        * **Made available as an injectable fixture**  
           _Just like a regular `@pytest.fixture`_
        
        ### 1. Set the stage to prepare for the test: `given_that`
        *    #### Simulate args passed via `apps.yaml` config
             See: [Appdaemon - Passing arguments to Apps](http://appdaemon.readthedocs.io/en/latest/APPGUIDE.html#passing-arguments-to-apps)
             ```python
             # Command
             given_that.passed_arg(ARG_KEY).is_set_to(ARG_VAL)
        
             # Example
             given_that.passed_arg('color').is_set_to('blue')
             ```
        
        *    #### State
             ```python
             # Command
             given_that.state_of(ENTITY_ID).is_set_to(STATE_TO_SET)
        
             # Example
             given_that.state_of(media_player.speaker).is_set_to('playing')
             ```
        
        *    #### Time
             ```python
             # Command
             given_that.time_is(TIME_AS_DATETIME)
        
             # Example
             given_that.time_is(time(hour=20))
             ```
        
        *    #### Extra
             ```python
             # Clear all calls recorded on the mocks
             given_that.mock_functions_are_cleared()
        
             # To also clear all mocked state, use the option: 'clear_mock_states'
             given_that.mock_functions_are_cleared(clear_mock_states=True)
        
             # To also clear all mocked passed args, use the option: 'clear_mock_passed_args'
             given_that.mock_functions_are_cleared(clear_mock_passed_args=True)
             ```
        
        ### 2. Trigger action on your automation
        The way Automations work in Appdaemon is: 
        * First you **register callback methods** during the `initialize()` phase
        * At some point **Appdaemon will trigger these callbacks**
        * Your Automation **reacts to the call on the callback**
        
        To **trigger actions** on your automation, simply **call one of the registered callbacks**.
        #### Example
        
        ##### `LivingRoomTest.py`
        ```python
        def test_during_night_light_turn_on(given_that, living_room, assert_that):
           ...
           living_room._new_motion(None, None, None)
           ...
        ```
        
        ##### With `LivingRoom.py`
        ```python
        class LivingRoom(hass.Hass):
            def initialize(self):
                ...
                self.listen_event(
                        self._new_motion, 
                        'motion',
                        entity_id='binary_sensor.bathroom_motion')
                ...
        
            def _new_motion(self, event_name, data, kwargs):
                < Handle motion here >
        ```
        
        > #### Note
        > It is best-practice to have an initial test that will test the callbacks
        > are _actually_ registered as expected during the `initialize()` phase.  
        >
        > _For now you need to use direct call to the mocked `hass_functions`_  
        > _See: [Full example — Kitchen](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/full_example/tests/test_kitchen.py#L41) & [Direct call to `hass_functions`](#direct-call-to-mocked-functions)_
        
        
        
        ### 3. Assert on your way out: `assert_that`
        
        *    #### Entities
             ```python
             # Available commmands
             assert_that(ENTITY_ID).was.turned_on(OPTIONAL_KWARGS)
             assert_that(ENTITY_ID).was.turned_off()
             assert_that(ENTITY_ID).was_not.turned_on(OPTIONAL_KWARGS)
             assert_that(ENTITY_ID).was_not.turned_off()
        
             # Examples
             assert_that('light.living_room').was.turned_on()
             assert_that('light.living_room').was.turned_on(color_name=SHOWER_COLOR)
             assert_that('light.living_room').was_not.turned_off()
             ```
        
        *    #### Services
             ```python
             # Available commmands
             assert_that(SERVICE).was.called_with(OPTIONAL_KWARGS)
             assert_that(SERVICE).was_not.called_with(OPTIONAL_KWARGS)
        
             # Examples
             assert_that('notify/pushbullet').was.called_with(
                                 message='Hello :)', 
                                 target='My Phone')
             
             assert_that('media_player/volume_set').was.called_with(
                                 entity_id='media_player.bathroom_speaker',
                                 volume_level=0.6)
             ```
        
        
        ### Bonus — Travel in Time: `time_travel`
        This helper simulate going forward in time.
        
        It will run the callbacks registered with the `run_in`function of Appdaemon:
        * **Order** is kept
        * Callback is run **only if due** at current simulated time
        * **Multiples calls** can be made in the same test
        * Automatically **resets between each test** _(with default config)_
        
        
         ```python
        # Available commmands
        
        ## Simulate time
        time_travel.fast_forward(MINUTES).minutes()
        time_travel.fast_forward(SECONDS).seconds()
        
        ## Assert time in test — Only useful for sanity check
        time_travel.assert_current_time(MINUTES).minutes()
        time_travel.assert_current_time(SECONDS).seconds()
        
        
        
        # Example
        
        # 2 services:
        #   * 'first/service': Should be called at T=3min
        #   * 'second/service': Should be called at T=5min
        time_travel.assert_current_time(0).minutes()
        
        time_travel.fast_forward(3).minutes()
        assert_that('some/service').was.called()
        assert_that('some_other/service').was_not.called()
        
        time_travel.fast_forward(2).minutes()
        assert_that('some_other/service').was.called()
        ```
        
        ---
        ## Examples
        ### Simple
        *  [**Pytest**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/pytest_example.py)
        *  [**Unittest**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/unittest_example.py)
        
        ### [Complete Project](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/tree/master/doc/full_example)
        * **Kitchen**
          * [**Automation**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/full_example/apps/kitchen.py)
          * [**Tests**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/full_example/tests/test_kitchen.py)
        * **Bathroom**
          * [**Automation**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/full_example/apps/bathroom.py)
          * [**Tests**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/full_example/tests/test_bathroom.py)
        
        ---
        
        ## Under The Hood
        > This section is **entirely optional**   
        > For a guide on how to use the framework, see the above sections!
        
        ### Understand the motivation
        **Why a test framework dedicated for Appdaemon?**  
        _The way Appdaemon allow the user to implement automations is based on inheritance.
        This makes testing not so trivial.
        This test framework abstracts away all that complexity, allowing for a smooth TDD experience._
        
        **Couldn't we just use the MVP pattern with clear interfaces at the boundaries?**  
        _Yes we could... but would we?  
        Let's be pragmatic, with this kind of project we're developing for our home, and we're a team of one.
        While being a huge proponent for [clean architecture](https://floriankempenich.com/post/11), I believe using such a complex architecture for such a simple project would only result in bringing more complexity than necessary._
        
        ### Enjoy the simplicity
        
        Every Automation in Appdaemon follows the same model:
        * **Inherit** from **`hass.Hass`**
        * **Call** Appdaemon API through **`self`**
        
        **AppdaemonTestFramework** captures all calls to the API and helpers make use of the information to implement common functionality needed in our tests.
        
        Methods from the `hass.Hass` class are patched globally, and injected in the helper classes.
        This is done with the `patch_hass()` wich returns a tuple containing:
        
        1. **`hass_functions`**: **dictionary** with all patched functions
        1. **`unpatch_callback`**: **callback** to un-patch all patched functions
        
        **`hass_functions`** are injected in the helpers when creating their instances.
        After all tests, **`unpatch_callback`** is used to un-patch all patched functions.
        
        Setup and teardown are handled in the [`conftest.py`](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/new-features/doc/full_example/conftest.py) file.
        
        
        #### Appdaemon Test Framework flow
        ###### 1. Setup
        * **Patch** `hass.Hass` functions
        * **Inject** `hass_functions` in helpers: `given_that`, `assert_that`, `time_travel`
        ###### 2. Test run
        * **Run** the test suite
        ###### 3. Teardown
        * **Un-patch** `hass.Hass` functions
        
        ### Feature focus
        #### `@automation_fixture`
        
        To be able to test an Appdaemon automation, the automation needs to be created **after** the framework has been 
        initialized. This could be done with a regular `@pytest.fixture` by injecting `given_that` (or any other helper) before 
        creating the automation. This would invoke the helper which would initialize the framework beforehand and the correct
        functions would be patched before the automation is created:
        ```python
        @pytest.fixture
        def living_room(given_that):
            # `None` dependencies are mocked by the framework
            living_room = LivingRoom(None, None, None, None, None, None, None, None)
            return living_room
        ```
        
        Also, for convenience, we could initialize the automation with its `initialize()` method to make it available in tests
        as it would be in production once Appdaemon is started, making sure the mocks are clear of any calls when the fixture 
        is injected:
        ```python
        @pytest.fixture
        def living_room(given_that):
            living_room = LivingRoom(None, None, None, None, None, None, None, None)
            living_room.initialize()
            given_that.mock_functions_are_cleared()
            return living_room
        ```
        
        However, since this code would have to be repeated for every single automation, it was creating un-necessary clutter.
        For that reason, the `@automation_fixture` was introduced. It is simple syntactic sugar, and performs the exact same 
        steps as the fixture above in much fewer lines:
        
        ```python
        @automation_fixture(LivingRoom)
        def living_room():
            pass
        ```
        
        
        ## Advanced Usage
        
        ### `@automation_fixture` - Extra features
        * #### Pre-initialization setup
          For some automations, the `initialize()` step requires some pre-configuration of the global state.
          Maybe it requires time to be setup, or maybe it needs some sensors to have a particular state. 
          Such pre-initialization setup is possible with the `@automation_fixture`. The fixture can be injected with the 
          following 2 arguments:
          * `given_that` _- For configuring the state_
          * `hass_functions` _- For more complex setup steps_
          
          Any code written in the fixture will be executed **before** initializing the automation. That way your
          `initialize()` function can safely rely on the Appdaemon framework and call some of its methods, all you 
          have to do is setup the context in the fixture.
          
          ##### Example
          Let's imagine an automation, `Terasse`, that turns on the light at night. During the `initialize()` phase, `Terasse`
          checks the current time to know if it should immediately turn on the light (for instance, if appdaemon is started 
          during the night).  
          Without mocking the time **before** the call to `initialize()` the test framework will not know which time to return 
          and an error will be raised. To prevent that, we set the time in the fixture, the code will be executed **before** the
          call to `initialize()` and no error will be raised.
          ```python
          @automation_fixture
          def terasse(given_that):
              given_that.time_is(datetime.time(hour=20))
              
              
          # With Terasse:
          class Terasse(hass.Hass):
              def initialize(self):
                  ...
                  current_time = self.time()
                  if self._is_during_night(current_time):
                      self._turn_on_light()
                  ...
          ```
        
        * #### Alternate versions
          The `@automation_fixture` can actually be used in 4 different ways:
          ```python
          # Single Class:               
          @automation_fixture(MyAutomation)
        
          # Multiple Classes:           
          @automation_fixture(MyAutomation, MyOtherAutomation)
        
          # Single Class w/ params:     
          @automation_fixture((upstairs.Bedroom, {'motion': 'binary_sensor.bedroom_motion'}))
        
          # Multiple Classes w/ params: 
          @automation_fixture(
            (upstairs.Bedroom, {'motion': 'binary_sensor.bedroom_motion'}),
            (upstairs.Bathroom, {'motion': 'binary_sensor.bathroom_motion'}),
          )
          ```
          
          The alternate versions can be useful for parametrized testing:
          * When multiple classes are passed, tests will be generated for each automation.   
          * When using parameters, the injected object will be a tuple: `(Initialized_Automation, params)`  
        
        ### Without `pytest`
        If you do no wish to use `pytest`, first maybe reconsider, `pytest` is awesome :)  
        If you're really set on using something else, worry not it's pretty straighforward too ;)
        
        What the [`conftest.py`](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/new-features/doc/full_example/conftest.py) file is doing is simply handling the setup & teardown, as well as providing the helpers as injectable fixtures.
        It is pretty easy to replicate the same behavior with your test framework of choice. For instance, with `unittest` a base `TestCase` can replace pytest [`conftest.py`](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/new-features/doc/full_example/conftest.py). See the [Unittest Example](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/doc/unittest_example.py)
        
        ### Direct call to mocked functions
        > /!\ WARNING — EXPERIMENTAL /!\
        
        **Want a functionality not implemented by the helpers?**  
        You can inject `hass_functions` directly in your tests, patched functions are `MagicMocks`.
        The list of patched functions can be found in the [**`init_framework` module**](https://github.com/FlorianKempenich/Appdaemon-Test-Framework/blob/master/appdaemontestframework/init_framework.py#L60).
        
        
        
        ---
        ## Contributing
        
        * **All contributions are welcome!**
        * **All PR must be accompanied with some tests showcasing the new feature**
        * **For new feature, a short discussion will take place to argue whether the feature makes sense and if this is the best place to implement**
        
        * **Then the process goes as follow:**  
          * Is readable? ✅
          * Passes the tests? ✅
          * I'm merging 🎉🎉
          
        
        > PR simply patching new functions in `hass_functions`
        >   * Will be merged almost immediately
        >   * Are exempt of the _"must have tests"_ rule, although there is rarely too much tests 😉
        
        Thanks for your contribution 😀 👍 
          
        
        ### Tests
        
        > **Note on the current state of tests:**  
        > The funny thing with this project is: it's the offspring of a test suite I was building as I worked on my home automation project. I ended up extracting these functions and published them online for others, like you, to use. Which means . . . most of the project itself doesn't have any unit tests 😮 That being said, there are _some_ tests.
        
        
        Two types of tests can be found:
        - Unit tests for recent features
        - Integration tests using the framework in a real-life scenario
        
        While it most likely isn't sufficient to guarantee a behavior coverage to fully rely on, it still provides some nice backwards compatility safety 🙂
        
        When adding new feature, you can TDD it, add unit tests later, or only rely on integration tests. Whichever way you go is totally fine for this project, but new features will need to have at least some sort of tests, even if they're super basic 🙂
        
        ---
        ## Author Information
        Follow me on Twitter: [@ThisIsFlorianK](https://twitter.com/ThisIsFlorianK)  
        Find out more about my work: [Florian Kempenich — Personal Website](https://floriankempenich.com)
        
Keywords: appdaemon homeassistant test tdd clean-code home-automation
Platform: UNKNOWN
Classifier: Environment :: Console
Classifier: Framework :: Pytest
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Topic :: Utilities
Classifier: Topic :: Home Automation
Classifier: Topic :: Software Development :: Testing
