From 22a86fe52eb6faf3aadee135472960dca1625b1d Mon Sep 17 00:00:00 2001 From: Till Date: Fri, 26 Jan 2024 21:10:39 +0100 Subject: [PATCH] Add Continuous Integration (#4) * Switch to Black formatting * Explicitly specify clang formatting * Add CI --- .clang-format | 9 ++ .github/workflows/ci.yaml | 122 ++++++++++++++++++ .pre-commit-config.yaml | 17 +++ .yamllint | 2 + components/desk_height_sensor/sensor.py | 26 ++-- .../desk_is_moving_sensor/binary_sensor.py | 11 +- components/desk_memory_button/button.py | 14 +- .../desk_memory_button/desk_memory_button.cpp | 62 ++++----- .../desk_memory_button/desk_memory_button.h | 2 +- components/desk_switch/switch.py | 19 +-- components/generic_desk/__init__.py | 41 +++--- components/generic_desk/generic_desk.h | 2 +- desk.yaml | 12 +- requirements.txt | 1 + tests/base.yaml | 24 ++++ tests/full.yaml | 69 ++++++++++ 16 files changed, 349 insertions(+), 84 deletions(-) create mode 100644 .clang-format create mode 100644 .github/workflows/ci.yaml create mode 100644 .pre-commit-config.yaml create mode 100644 .yamllint create mode 100644 requirements.txt create mode 100644 tests/base.yaml create mode 100644 tests/full.yaml diff --git a/.clang-format b/.clang-format new file mode 100644 index 0000000..a565c3b --- /dev/null +++ b/.clang-format @@ -0,0 +1,9 @@ +UseTab: Never +IndentWidth: 4 +NamespaceIndentation: All +AccessModifierOffset: -4 +BreakBeforeBraces: Allman +AllowShortIfStatementsOnASingleLine: false +IndentCaseLabels: false +SortIncludes: false +ColumnLimit: 0 diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml new file mode 100644 index 0000000..99c329b --- /dev/null +++ b/.github/workflows/ci.yaml @@ -0,0 +1,122 @@ +name: CI + +# yamllint disable-line rule:truthy +on: + push: + branches: [main, dev] + + pull_request: + merge_group: + +permissions: + contents: read + +concurrency: + # yamllint disable-line rule:line-length + group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} + cancel-in-progress: true + +jobs: + ci: + name: ${{ matrix.name }} + runs-on: ubuntu-latest + strategy: + fail-fast: false + max-parallel: 5 + matrix: + include: + - id: test + file: tests/full.yaml + name: Test tests/full.yaml + pio_cache_key: full + - id: test + file: tests/base.yaml + name: Test tests/base.yaml + pio_cache_key: base + - id: clang-format + name: Run clang-format + - id: yamllint + name: Run yamllint + - id: black-format + name: Run black-format + - id: isort + name: Run isort + + steps: + - uses: actions/checkout@v3 + - name: Set up Python + uses: actions/setup-python@v4 + id: python + with: + python-version: "3.9" + + - name: Cache virtualenv + uses: actions/cache@v3 + with: + path: .venv + # yamllint disable-line rule:line-length + key: venv-${{ steps.python.outputs.python-version }} + restore-keys: | + venv-${{ steps.python.outputs.python-version }} + - name: Set up virtualenv + # yamllint disable rule:line-length + run: | + python -m venv .venv + source .venv/bin/activate + pip install -U pip + pip install -r requirements.txt + echo "$GITHUB_WORKSPACE/.venv/bin" >> $GITHUB_PATH + echo "VIRTUAL_ENV=$GITHUB_WORKSPACE/.venv" >> $GITHUB_ENV + # yamllint enable rule:line-length + + # Use per check platformio cache because checks use different parts + - name: Cache platformio + uses: actions/cache@v3 + with: + path: ~/.platformio + # yamllint disable-line rule:line-length + key: platformio-${{ matrix.pio_cache_key }}-${{ hashFiles('platformio.ini') }} + if: matrix.id == 'test' + + - run: esphome compile ${{ matrix.file }} + if: matrix.id == 'test' + env: + # Also cache libdeps, store them in a ~/.platformio subfolder + PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps + + - name: Run clang-format + uses: jidicula/clang-format-action@v4.11.0 + with: + clang-format-version: "13" + check-path: "components" + if: matrix.id == 'clang-format' + + - name: Run yamllint + if: matrix.id == 'yamllint' + uses: frenck/action-yamllint@v1.4.0 + + - name: Run black-format + if: matrix.id == 'black-format' + uses: psf/black@stable + with: + options: "--check --verbose" + version: "~= 24.1" + + - name: Run isort + if: matrix.id == 'isort' + uses: isort/isort-action@master + with: + requirementsFiles: "requirements.txt" + + ci-status: + name: CI Status + runs-on: ubuntu-latest + needs: [ci] + if: always() + steps: + - name: Successful deploy + if: ${{ !(contains(needs.*.result, 'failure')) }} + run: exit 0 + - name: Failing deploy + if: ${{ contains(needs.*.result, 'failure') }} + run: exit 1 diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 0000000..5763f0d --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,17 @@ +repos: + - repo: https://github.com/psf/black-pre-commit-mirror + rev: 24.1.0 + hooks: + - id: black + language_version: python3.11 + + - repo: https://github.com/pycqa/isort + rev: 5.13.2 + hooks: + - id: isort + name: isort (python) + + - repo: https://github.com/pre-commit/mirrors-clang-format + rev: v17.0.6 + hooks: + - id: clang-format \ No newline at end of file diff --git a/.yamllint b/.yamllint new file mode 100644 index 0000000..e2da62b --- /dev/null +++ b/.yamllint @@ -0,0 +1,2 @@ +ignore: | + venv/ \ No newline at end of file diff --git a/components/desk_height_sensor/sensor.py b/components/desk_height_sensor/sensor.py index f78964a..8d5d4aa 100644 --- a/components/desk_height_sensor/sensor.py +++ b/components/desk_height_sensor/sensor.py @@ -2,18 +2,28 @@ import esphome.config_validation as cv from esphome.components import sensor from esphome.const import CONF_ID, ICON_ARROW_EXPAND_VERTICAL -from ..generic_desk import GenericDesk, CONF_DESK_ID -DEPENDENCIES = ['generic_desk'] +from ..generic_desk import CONF_DESK_ID, GenericDesk -generic_desk_sensor_ns = cg.esphome_ns.namespace('generic_desk') +DEPENDENCIES = ["generic_desk"] + +generic_desk_sensor_ns = cg.esphome_ns.namespace("generic_desk") GenericDeskSensor = generic_desk_sensor_ns.class_( - 'GenericDesk', sensor.Sensor, cg.PollingComponent) + "GenericDesk", sensor.Sensor, cg.PollingComponent +) -CONFIG_SCHEMA = sensor.sensor_schema(unit_of_measurement="cm", icon=ICON_ARROW_EXPAND_VERTICAL, accuracy_decimals=1).extend({ - cv.GenerateID(): cv.declare_id(GenericDeskSensor), - cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk) -}).extend(cv.COMPONENT_SCHEMA) +CONFIG_SCHEMA = ( + sensor.sensor_schema( + unit_of_measurement="cm", icon=ICON_ARROW_EXPAND_VERTICAL, accuracy_decimals=1 + ) + .extend( + { + cv.GenerateID(): cv.declare_id(GenericDeskSensor), + cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk), + } + ) + .extend(cv.COMPONENT_SCHEMA) +) def to_code(config): diff --git a/components/desk_is_moving_sensor/binary_sensor.py b/components/desk_is_moving_sensor/binary_sensor.py index b853c16..2bc9195 100644 --- a/components/desk_is_moving_sensor/binary_sensor.py +++ b/components/desk_is_moving_sensor/binary_sensor.py @@ -1,14 +1,15 @@ import esphome.codegen as cg import esphome.config_validation as cv from esphome.components import binary_sensor -from ..generic_desk import GenericDesk, CONF_DESK_ID -DEPENDENCIES = ['generic_desk'] +from ..generic_desk import CONF_DESK_ID, GenericDesk +DEPENDENCIES = ["generic_desk"] -CONFIG_SCHEMA = binary_sensor.binary_sensor_schema().extend({ - cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk) -}) + +CONFIG_SCHEMA = binary_sensor.binary_sensor_schema().extend( + {cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk)} +) async def to_code(config): diff --git a/components/desk_memory_button/button.py b/components/desk_memory_button/button.py index 7893a58..4d5af44 100644 --- a/components/desk_memory_button/button.py +++ b/components/desk_memory_button/button.py @@ -2,21 +2,21 @@ import esphome.config_validation as cv from esphome.components import button from esphome.const import CONF_ID -from ..generic_desk import GenericDesk, CONF_DESK_ID -DEPENDENCIES = ['generic_desk'] +from ..generic_desk import CONF_DESK_ID, GenericDesk -CONF_MEMORY_BUTTON_ID = 'memory_id' +DEPENDENCIES = ["generic_desk"] -memory_button_ns = cg.esphome_ns.namespace('memory_button') -MemoryButton = memory_button_ns.class_( - "MemoryButton", button.Button, cg.Component) +CONF_MEMORY_BUTTON_ID = "memory_id" + +memory_button_ns = cg.esphome_ns.namespace("memory_button") +MemoryButton = memory_button_ns.class_("MemoryButton", button.Button, cg.Component) CONFIG_SCHEMA = button.BUTTON_SCHEMA.extend( { cv.GenerateID(): cv.declare_id(MemoryButton), cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk), - cv.Required(CONF_MEMORY_BUTTON_ID): cv.int_ + cv.Required(CONF_MEMORY_BUTTON_ID): cv.int_, } ).extend(cv.COMPONENT_SCHEMA) diff --git a/components/desk_memory_button/desk_memory_button.cpp b/components/desk_memory_button/desk_memory_button.cpp index c6c82cb..a6ad294 100644 --- a/components/desk_memory_button/desk_memory_button.cpp +++ b/components/desk_memory_button/desk_memory_button.cpp @@ -3,42 +3,42 @@ namespace esphome { - namespace memory_button - { + namespace memory_button + { - static const char *const TAG = "Desk.memoryButton"; + static const char *const TAG = "Desk.memoryButton"; - void MemoryButton::dump_config() - { - LOG_BUTTON("", "Memory Button", this); - } + void MemoryButton::dump_config() + { + LOG_BUTTON("", "Memory Button", this); + } - void MemoryButton::press_action() - { - for (unsigned int i = 0; i <= MESSAGE_REPETITIONS; i++) - { - switch (this->memory_id) + void MemoryButton::press_action() { - case 0: - uart_device->write_array({0xea, 0xf5, 0x41, 0x00, 0x00, 0x41, 0xd4, 0x28}); - break; - case 1: - uart_device->write_array({0xea, 0xf5, 0x42, 0x00, 0x00, 0x42, 0x94, 0x6d}); - break; - case 2: - uart_device->write_array({0xea, 0xf5, 0x43, 0x00, 0x00, 0x43, 0x54, 0x51}); - break; - case 3: - uart_device->write_array({0xea, 0xf5, 0x44, 0x00, 0x00, 0x44, 0x14, 0xe7}); - break; + for (unsigned int i = 0; i <= MESSAGE_REPETITIONS; i++) + { + switch (this->memory_id) + { + case 0: + uart_device->write_array({0xea, 0xf5, 0x41, 0x00, 0x00, 0x41, 0xd4, 0x28}); + break; + case 1: + uart_device->write_array({0xea, 0xf5, 0x42, 0x00, 0x00, 0x42, 0x94, 0x6d}); + break; + case 2: + uart_device->write_array({0xea, 0xf5, 0x43, 0x00, 0x00, 0x43, 0x54, 0x51}); + break; + case 3: + uart_device->write_array({0xea, 0xf5, 0x44, 0x00, 0x00, 0x44, 0x14, 0xe7}); + break; - default: - break; + default: + break; + } + uart_device->flush(); + delay(5); + } } - uart_device->flush(); - delay(5); - } - } - } // namespace output + } // namespace memory_button } // namespace esphome \ No newline at end of file diff --git a/components/desk_memory_button/desk_memory_button.h b/components/desk_memory_button/desk_memory_button.h index ff56bc1..8b1ef59 100644 --- a/components/desk_memory_button/desk_memory_button.h +++ b/components/desk_memory_button/desk_memory_button.h @@ -24,5 +24,5 @@ namespace esphome uart::UARTDevice *uart_device; }; - } // namespace output + } // namespace memory_button } // namespace esphome \ No newline at end of file diff --git a/components/desk_switch/switch.py b/components/desk_switch/switch.py index 96a313a..bf66c6f 100644 --- a/components/desk_switch/switch.py +++ b/components/desk_switch/switch.py @@ -2,18 +2,21 @@ import esphome.config_validation as cv from esphome.components import switch from esphome.const import CONF_ID -from ..generic_desk import GenericDesk, CONF_DESK_ID + +from ..generic_desk import CONF_DESK_ID, GenericDesk CONF_DIRECTION = "direction" -desk_switch_ns = cg.esphome_ns.namespace('desk_switch') -DeskSwitch = desk_switch_ns.class_('DeskSwitch', switch.Switch, cg.Component) +desk_switch_ns = cg.esphome_ns.namespace("desk_switch") +DeskSwitch = desk_switch_ns.class_("DeskSwitch", switch.Switch, cg.Component) -CONFIG_SCHEMA = switch.SWITCH_SCHEMA.extend({ - cv.GenerateID(): cv.declare_id(DeskSwitch), - cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk), - cv.Required(CONF_DIRECTION): cv.boolean -}).extend(cv.COMPONENT_SCHEMA) +CONFIG_SCHEMA = switch.SWITCH_SCHEMA.extend( + { + cv.GenerateID(): cv.declare_id(DeskSwitch), + cv.GenerateID(CONF_DESK_ID): cv.use_id(GenericDesk), + cv.Required(CONF_DIRECTION): cv.boolean, + } +).extend(cv.COMPONENT_SCHEMA) async def to_code(config): diff --git a/components/generic_desk/__init__.py b/components/generic_desk/__init__.py index b1a0e5b..cbc8069 100644 --- a/components/generic_desk/__init__.py +++ b/components/generic_desk/__init__.py @@ -1,29 +1,38 @@ import esphome.codegen as cg import esphome.config_validation as cv from esphome.components import uart -from esphome.components.uart import UARTComponent, CONF_UART_ID +from esphome.components.uart import CONF_UART_ID, UARTComponent from esphome.const import CONF_ID -DEPENDENCIES = ['uart'] -AUTO_LOAD = ['desk_memory_button', 'desk_switch', 'desk_height_sensor', - 'desk_is_moving_sensor', 'sensor', 'binary_sensor', 'button', 'switch'] +DEPENDENCIES = ["uart"] +AUTO_LOAD = [ + "desk_memory_button", + "desk_switch", + "desk_height_sensor", + "desk_is_moving_sensor", + "sensor", + "binary_sensor", + "button", + "switch", +] MULTI_CONF = True -CONF_DESK_ID = 'desk_id' -CONF_BASE_HEIGHT = 'base_height' -CONF_CORRECTION_TERM = 'correction_term' +CONF_DESK_ID = "desk_id" +CONF_BASE_HEIGHT = "base_height" +CONF_CORRECTION_TERM = "correction_term" -generic_desk_ns = cg.esphome_ns.namespace('generic_desk') -GenericDesk = generic_desk_ns.class_( - 'GenericDesk', cg.Component, uart.UARTDevice) +generic_desk_ns = cg.esphome_ns.namespace("generic_desk") +GenericDesk = generic_desk_ns.class_("GenericDesk", cg.Component, uart.UARTDevice) -CONFIG_SCHEMA = uart.UART_DEVICE_SCHEMA.extend({ - cv.GenerateID(): cv.declare_id(GenericDesk), - cv.Required(CONF_UART_ID): cv.use_id(UARTComponent), - cv.Required(CONF_BASE_HEIGHT): cv.float_, - cv.Optional(CONF_CORRECTION_TERM, 1.0): cv.float_ -}).extend(cv.polling_component_schema('250ms')) +CONFIG_SCHEMA = uart.UART_DEVICE_SCHEMA.extend( + { + cv.GenerateID(): cv.declare_id(GenericDesk), + cv.Required(CONF_UART_ID): cv.use_id(UARTComponent), + cv.Required(CONF_BASE_HEIGHT): cv.float_, + cv.Optional(CONF_CORRECTION_TERM, 1.0): cv.float_, + } +).extend(cv.polling_component_schema("250ms")) async def to_code(config): diff --git a/components/generic_desk/generic_desk.h b/components/generic_desk/generic_desk.h index 125c8b8..72316ab 100644 --- a/components/generic_desk/generic_desk.h +++ b/components/generic_desk/generic_desk.h @@ -71,5 +71,5 @@ namespace esphome float last_height = 0.0; }; - } // namespace empty_sensor + } // namespace generic_desk } // namespace esphome diff --git a/desk.yaml b/desk.yaml index c00b051..d46cb5d 100644 --- a/desk.yaml +++ b/desk.yaml @@ -48,10 +48,10 @@ sensor: name: "Desk Height" binary_sensor: - - platform: desk_is_moving_sensor - desk_id: desk1 - id: desk_is_moving - name: "Desk is Moving" + - platform: desk_is_moving_sensor + desk_id: desk1 + id: desk_is_moving + name: "Desk is Moving" switch: - platform: desk_switch @@ -62,7 +62,7 @@ switch: - platform: desk_switch desk_id: desk1 name: "Down" - direction: false + direction: false button: - platform: desk_memory_button @@ -84,5 +84,3 @@ button: desk_id: desk1 name: "M4" memory_id: 4 - - diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..c3c0b83 --- /dev/null +++ b/requirements.txt @@ -0,0 +1 @@ +esphome \ No newline at end of file diff --git a/tests/base.yaml b/tests/base.yaml new file mode 100644 index 0000000..53eaeb5 --- /dev/null +++ b/tests/base.yaml @@ -0,0 +1,24 @@ +esphome: + name: desk + +esp8266: + board: esp01_1m + +external_components: + - source: + type: local + path: ../components + +logger: + baud_rate: 0 + +uart: + id: uart_bus + tx_pin: TX + rx_pin: RX + baud_rate: 57600 + +generic_desk: + id: desk1 + uart_id: uart_bus + base_height: 66.4 diff --git a/tests/full.yaml b/tests/full.yaml new file mode 100644 index 0000000..a0cfdda --- /dev/null +++ b/tests/full.yaml @@ -0,0 +1,69 @@ +esphome: + name: desk + +esp8266: + board: esp01_1m + +external_components: + - source: + type: local + path: ../components + +logger: + baud_rate: 0 + +uart: + id: uart_bus + tx_pin: TX + rx_pin: RX + baud_rate: 57600 + +generic_desk: + id: desk1 + uart_id: uart_bus + base_height: 66.4 + correction_term: 1.24836 + +sensor: + - platform: desk_height_sensor + desk_id: desk1 + id: desk_height + name: "Desk Height" + +binary_sensor: + - platform: desk_is_moving_sensor + desk_id: desk1 + id: desk_is_moving + name: "Desk is Moving" + +switch: + - platform: desk_switch + desk_id: desk1 + name: "Up" + direction: true + + - platform: desk_switch + desk_id: desk1 + name: "Down" + direction: false + +button: + - platform: desk_memory_button + desk_id: desk1 + name: "M1" + memory_id: 1 + + - platform: desk_memory_button + desk_id: desk1 + name: "M2" + memory_id: 2 + + - platform: desk_memory_button + desk_id: desk1 + name: "M3" + memory_id: 3 + + - platform: desk_memory_button + desk_id: desk1 + name: "M4" + memory_id: 4