diff --git a/CHANGELOG.md b/CHANGELOG.md index afee0419e..0f8ccf418 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,7 +2,10 @@ ### Features + - Add ability to use Actions in Custom Functions - Add Input Action: Execute Python 3 Code ([#1334](https://github.com/kizniche/Mycodo/issues/1334)) + - Add Function: Adafruit Neokey (Key Press Executes Actions) ([#1353](https://github.com/kizniche/Mycodo/issues/1353)) + - Change deprecated threading.currentThread to threading.current_thread ### Bugfixes diff --git a/mycodo/functions/function_adafruit_neokey_01.py b/mycodo/functions/function_adafruit_neokey_01.py new file mode 100644 index 000000000..10496d30f --- /dev/null +++ b/mycodo/functions/function_adafruit_neokey_01.py @@ -0,0 +1,351 @@ +# coding=utf-8 +# +# function_adafruit_neokey_01.py - Function to utilize the Adafruit Neokey +# +# Copyright (C) 2015-2023 Kyle T. Gabriel +# +# This file is part of Mycodo +# +# Mycodo is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Mycodo is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Mycodo. If not, see . +# +# Contact at kylegabriel.com +# +import threading +import time +import traceback + +from mycodo.config_translations import TRANSLATIONS +from mycodo.databases.models import Actions +from mycodo.databases.models import CustomController +from mycodo.databases.models import FunctionChannel +from mycodo.functions.base_function import AbstractFunction +from mycodo.mycodo_client import DaemonControl +from mycodo.utils.database import db_retrieve_table_daemon +from mycodo.utils.influx import write_influxdb_value + +# Measurements +measurements_dict = { + 0: { + 'measurement': 'boolean', + 'unit': 'bool' + }, + 1: { + 'measurement': 'boolean', + 'unit': 'bool' + }, + 2: { + 'measurement': 'boolean', + 'unit': 'bool' + }, + 3: { + 'measurement': 'boolean', + 'unit': 'bool' + } +} + +channels_dict = { + 0: { + 'name': 'Key 1', + 'measurements': [0] + }, + 1: { + 'name': 'Key 2', + 'measurements': [1] + }, + 2: { + 'name': 'Key 3', + 'measurements': [2] + }, + 3: { + 'name': 'Key 4', + 'measurements': [3] + } +} + + +def execute_at_modification( + messages, + mod_function, + request_form, + custom_options_dict_presave, + custom_options_channels_dict_presave, + custom_options_dict_postsave, + custom_options_channels_dict_postsave): + """ + This function allows you to view and modify the output and channel settings when the user clicks + save on the user interface. Both the output and channel settings are passed to this function, as + dictionaries. Additionally, both the pre-saved and post-saved options are available, as it's + sometimes useful to know what settings changed and from what values. You can modify the post-saved + options and these will be stored in the database. + :param mod_function: The post-saved output database entry, minus the custom_options settings + :param request_form: The requests.form object the user submitted + :param custom_options_dict_presave: dict of pre-saved custom output options + :param custom_options_channels_dict_presave: dict of pre-saved custom output channel options + :param custom_options_dict_postsave: dict of post-saved custom output options + :param custom_options_channels_dict_postsave: dict of post-saved custom output channel options + :return: + """ + page_refresh = False + + try: + pass # TODO: Add check for properly-formatted Action ID strings + except Exception: + messages["error"].append("execute_at_modification() Error: {}".format(traceback.print_exc())) + + return (messages, + mod_function, + custom_options_dict_postsave, + custom_options_channels_dict_postsave, + page_refresh) + + +FUNCTION_INFORMATION = { + 'function_name_unique': 'function_adafruit_neokey_01', + 'function_name': 'Adafruit Neokey (Key Executes Actions)', + 'function_manufacturer': 'Adafruit', + 'function_library': 'adafruit-circuitpython-neokey', + 'execute_at_modification': execute_at_modification, + 'measurements_dict': measurements_dict, + 'channels_dict': channels_dict, + + 'message': 'This Function executes actions when a key is pressed. Add actions at the bottom of this module, then enter one or more short action IDs for each key, separated by commas. The Action ID is found next to the Action (for example, the Action "[Action 0559689e] Controller: Activate" has an Action ID of 0559689e. When entering Action ID(s), separate multiple IDs by commas (for example, "asdf1234" or "asdf1234,qwer5678,zxcv0987"). Actions will be executed in the order they are entered in the text string.', + + 'dependencies_module': [ + ('pip-pypi', 'usb.core', 'pyusb==1.1.1'), + ('pip-pypi', 'adafruit_extended_bus', 'Adafruit-extended-bus==1.0.2'), + ('pip-pypi', 'adafruit_neokey', 'adafruit-circuitpython-neokey==1.1.2') + ], + + 'options_disabled': [ + 'measurements_select', + 'measurements_configure' + ], + + 'options_enabled': [ + 'custom_options', + 'enable_actions' + ], + + 'custom_options': [ + { + 'id': 'i2c_address', + 'type': 'text', + 'default_value': '0x30', + 'required': True, + 'name': TRANSLATIONS['i2c_location']['title'], + 'phrase': '' + }, + { + 'id': 'i2c_bus', + 'type': 'integer', + 'default_value': 1, + 'required': True, + 'name': TRANSLATIONS['i2c_bus']['title'], + 'phrase': '' + }, + { + 'id': 'key_led_brightness', + 'type': 'float', + 'default_value': 0.2, + 'required': True, + 'name': 'LED Brightness (0.0-1.0)', + 'phrase': 'The brightness of the LEDs' + }, + ], + + 'custom_channel_options': [ + { + 'id': 'name', + 'type': 'text', + 'default_value': '', + 'required': False, + 'name': TRANSLATIONS['name']['title'], + 'phrase': TRANSLATIONS['name']['phrase'] + }, + { + 'id': 'key_action_ids', + 'type': 'text', + 'default_value': '', + 'required': False, + 'name': 'Action ID(s)', + 'phrase': 'Set which action(s) execute when the key is pressed. Enter one or more Action IDs, separated by commas' + }, + { + 'id': 'key_led_delay', + 'type': 'float', + 'default_value': 1.5, + 'required': True, + 'name': 'LED Delay', + 'phrase': 'How long to leave the LED on after the last action executes.' + }, + { + 'type': 'message', + 'default_value': 'The LED color of the key before being pressed, after being pressed, and when the last action is running. Values use colorwheel, with 85 as green, 170 as blue, and 255 as red, with the values between being the rest of the rainbow. Setting to 0 will turn the LED off.', + }, + { + 'id': 'led_rest', + 'type': 'integer', + 'default_value': 0, + 'required': True, + 'name': 'Resting LED Color (0-255)', + 'phrase': 'The LED color while no actions are running' + }, + { + 'id': 'led_start', + 'type': 'integer', + 'default_value': 85, + 'required': True, + 'name': 'Actions Running LED Color: (0-255)', + 'phrase': 'The LED color while all but the last action is running' + }, + { + 'id': 'led_last', + 'type': 'integer', + 'default_value': 255, + 'required': True, + 'name': 'Last Action LED Color (0-255)', + 'phrase': 'The LED color while the last action is running' + } + ] +} + + +class CustomModule(AbstractFunction): + """ + Class to operate custom controller + """ + def __init__(self, function, testing=False): + super().__init__(function, testing=testing, name=__name__) + + self.options_channels = {} + self.neokey = None + self.colorwheel = None + self.control = None + + # Initialize custom options + self.i2c_address = None + self.i2c_bus = None + self.key_led_brightness = None + + # Set custom options + custom_function = db_retrieve_table_daemon( + CustomController, unique_id=self.unique_id) + self.setup_custom_options( + FUNCTION_INFORMATION['custom_options'], custom_function) + + if not testing: + self.try_initialize() + + def initialize(self): + from adafruit_neokey.neokey1x4 import NeoKey1x4 + from adafruit_extended_bus import ExtendedI2C + from rainbowio import colorwheel + + self.colorwheel = colorwheel + self.control = DaemonControl() + + try: + function_channels = db_retrieve_table_daemon( + FunctionChannel).filter(FunctionChannel.function_id == self.unique_id).all() + self.options_channels = self.setup_custom_channel_options_json( + FUNCTION_INFORMATION['custom_channel_options'], function_channels) + + for channel in range(4): + list_action_ids = [] + if self.options_channels['key_action_ids'][channel].replace(" ", ""): + for action_id in self.options_channels['key_action_ids'][channel].replace(" ", "").split(","): + list_action_ids.append(action_id) + self.options_channels['key_action_ids'][channel] = list_action_ids + self.logger.debug(f"Parsed Action IDs: {self.options_channels['key_action_ids']}") + except: + self.logger.exception("Parsing action IDs") + + self.logger.debug(f"Brightness: {self.key_led_brightness}") + self.logger.debug(f"Colors: " + f"{self.options_channels['led_rest']}, " + f"{self.options_channels['led_start']}, " + f"{self.options_channels['led_last']}") + + try: + self.neokey = NeoKey1x4( + ExtendedI2C(self.i2c_bus), + addr=int(str(self.i2c_address), 16)) + + self.neokey.pixels.brightness = self.key_led_brightness + + for key in range(4): + self.set_color(key, self.options_channels['led_rest'][key]) + except: + self.logger.exception("Initializing device") + + def listener(self): + """This function will be turned into a thread to watch for key presses.""" + while self.running: + for key in range(4): + if self.neokey[key]: + self.logger.debug(f"Key {key + 1} Pressed") + self.set_color(key, self.options_channels['led_start'][key]) + + while self.neokey[key] and self.running: + time.sleep(0.2) + + write_influxdb_value( + self.unique_id, + self.channels_measurement[key].unit, + value=1, + measure=self.channels_measurement[key].measurement, + channel=key) + + key_thread = threading.Thread( + target=self.run_key_actions, + args=(key,)) + key_thread.start() + time.sleep(0.25) + + def run_key_actions(self, key): + for i, each_id in enumerate(self.options_channels['key_action_ids'][key]): + if i == len(self.options_channels['key_action_ids'][key]) - 1: + self.set_color(key, self.options_channels['led_last'][key]) + + if not self.running: + self.logger.info("Detected shutting down, stopping execution of Actions") + self.set_color(key, self.options_channels['led_rest'][key]) + return + + action = db_retrieve_table_daemon(Actions).filter( + Actions.unique_id.startswith(each_id)).first() + + if not action: + self.logger.error(f"Unknown Action ID: '{each_id}'") + continue + + try: + self.logger.debug(f"Executing Action with ID {each_id}") + return_dict = self.control.trigger_action( + action.unique_id, + value={"message": "", "measurements_dict": {}}, + debug=False) + if return_dict and "message" in return_dict: + self.logger.debug(f"Action {each_id} return message: {return_dict['message']}") + except: + self.logger.exception(f"Executing Action with ID {each_id}") + + time.sleep(self.options_channels['key_led_delay'][key]) + self.set_color(key, self.options_channels['led_rest'][key]) + + def set_color(self, key, color): + if color == 0: + self.neokey.pixels[key] = 0x0 # LED off + else: + self.neokey.pixels[key] = self.colorwheel(color) diff --git a/mycodo/mycodo_flask/templates/pages/function_options/custom_function_options.html b/mycodo/mycodo_flask/templates/pages/function_options/custom_function_options.html index db35a3734..c5d342ba6 100644 --- a/mycodo/mycodo_flask/templates/pages/function_options/custom_function_options.html +++ b/mycodo/mycodo_flask/templates/pages/function_options/custom_function_options.html @@ -265,6 +265,10 @@
{{_('Measurement Settings')}}
{% if 'enable_actions' in dict_options['options_enabled']%} +
+
{{_('Actions')}}
+
+
{{form_actions.csrf_token}} {{form_actions.device_id(class_='form-control', value=each_function.unique_id)}} diff --git a/mycodo/outputs/on_off_neopixel_rgb.py b/mycodo/outputs/on_off_neopixel_rgb.py index 0ea601ad0..26c7466b6 100644 --- a/mycodo/outputs/on_off_neopixel_rgb.py +++ b/mycodo/outputs/on_off_neopixel_rgb.py @@ -4,7 +4,7 @@ # import time from threading import Thread -from threading import currentThread +from threading import current_thread from flask_babel import lazy_gettext @@ -408,7 +408,7 @@ def go_to_color(self, t, current_color, new_color): return current_color def rainbow(self): - t = currentThread() + t = current_thread() maximum = self.rainbow_brightness cycle = [ (maximum, 0, 0), # Red diff --git a/mycodo/outputs/on_off_neopixel_rgb_spi.py b/mycodo/outputs/on_off_neopixel_rgb_spi.py index 745c53e4b..49648a9f8 100644 --- a/mycodo/outputs/on_off_neopixel_rgb_spi.py +++ b/mycodo/outputs/on_off_neopixel_rgb_spi.py @@ -4,7 +4,7 @@ # import time from threading import Thread -from threading import currentThread +from threading import current_thread from flask_babel import lazy_gettext @@ -368,7 +368,7 @@ def go_to_color(self, t, current_color, new_color): return current_color def rainbow(self): - t = currentThread() + t = current_thread() maximum = self.rainbow_brightness cycle = [ (maximum, 0, 0), # Red