more complexity in the test cases
This commit is contained in:
@@ -1,312 +0,0 @@
|
|||||||
{
|
|
||||||
"cells": [
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 6,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": [
|
|
||||||
"import os\n",
|
|
||||||
"import sys\n",
|
|
||||||
"import json\n",
|
|
||||||
"from pathlib import Path\n",
|
|
||||||
"\n",
|
|
||||||
"import yaml\n",
|
|
||||||
"from rich import print"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 2,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": [
|
|
||||||
"class EmptyClass:\n",
|
|
||||||
" pass\n",
|
|
||||||
"\n",
|
|
||||||
"self = EmptyClass()\n",
|
|
||||||
"self.AD = EmptyClass()\n",
|
|
||||||
"self.AD.app_dir = '/mnt/ad_dev_conf/apps'"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 3,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">Python parents:\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"Python parents:\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/my_repo/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_pkg</span>\n",
|
|
||||||
" - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">apps</span>\n",
|
|
||||||
" - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/my_repo/my_pkg/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_sub_pkg</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/my_repo/\u001b[0m\u001b[95mmy_pkg\u001b[0m\n",
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/\u001b[0m\u001b[95mapps\u001b[0m\n",
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/my_repo/my_pkg/\u001b[0m\u001b[95mmy_sub_pkg\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">Module parents:\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"Module parents:\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">apps</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/\u001b[0m\u001b[95mapps\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">Package parents:\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"Package parents:\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/my_repo/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_pkg</span>\n",
|
|
||||||
" - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/my_repo/my_pkg/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_sub_pkg</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/my_repo/\u001b[0m\u001b[95mmy_pkg\u001b[0m\n",
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/my_repo/my_pkg/\u001b[0m\u001b[95mmy_sub_pkg\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">Top package parents:\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"Top package parents:\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/my_repo/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_pkg</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/my_repo/\u001b[0m\u001b[95mmy_pkg\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">Imports\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"Imports\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">apps</span>\n",
|
|
||||||
" - <span style=\"color: #800080; text-decoration-color: #800080\">/mnt/ad_dev_conf/apps/</span><span style=\"color: #ff00ff; text-decoration-color: #ff00ff\">my_repo</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/\u001b[0m\u001b[95mapps\u001b[0m\n",
|
|
||||||
" - \u001b[35m/mnt/ad_dev_conf/apps/\u001b[0m\u001b[95mmy_repo\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"base_path = Path(self.AD.app_dir)\n",
|
|
||||||
"\n",
|
|
||||||
"python_file_parents = set(\n",
|
|
||||||
" f.parent.resolve()\n",
|
|
||||||
" for f in base_path.rglob('*.py')\n",
|
|
||||||
")\n",
|
|
||||||
"print('Python parents:')\n",
|
|
||||||
"print('\\n'.join(f' - {p.as_posix()}' for p in python_file_parents))\n",
|
|
||||||
"\n",
|
|
||||||
"module_parents = set(\n",
|
|
||||||
" p for p in python_file_parents\n",
|
|
||||||
" if not (p / '__init__.py').exists()\n",
|
|
||||||
")\n",
|
|
||||||
"print('Module parents:')\n",
|
|
||||||
"print('\\n'.join(f' - {p.as_posix()}' for p in module_parents))\n",
|
|
||||||
"\n",
|
|
||||||
"package_parents = set(\n",
|
|
||||||
" p for p in python_file_parents\n",
|
|
||||||
" if (p / '__init__.py').exists()\n",
|
|
||||||
")\n",
|
|
||||||
"print('Package parents:')\n",
|
|
||||||
"print('\\n'.join(f' - {p.as_posix()}' for p in package_parents))\n",
|
|
||||||
"\n",
|
|
||||||
"top_packages_dirs = set(\n",
|
|
||||||
" p for p in package_parents\n",
|
|
||||||
" if not (p.parent / '__init__.py').exists()\n",
|
|
||||||
")\n",
|
|
||||||
"print('Top package parents:')\n",
|
|
||||||
"print('\\n'.join(f' - {p.as_posix()}' for p in top_packages_dirs))\n",
|
|
||||||
"\n",
|
|
||||||
"import_dirs = module_parents | set(p.parent for p in top_packages_dirs)\n",
|
|
||||||
"print('Imports')\n",
|
|
||||||
"print('\\n'.join(f' - {p.as_posix()}' for p in import_dirs))"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 4,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/html": [
|
|
||||||
"<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"font-weight: bold\">{</span>\n",
|
|
||||||
" <span style=\"color: #800080; text-decoration-color: #800080; font-weight: bold\">PosixPath</span><span style=\"font-weight: bold\">(</span><span style=\"color: #008000; text-decoration-color: #008000\">'/mnt/ad_dev_conf/apps/my_repo/my_pkg/__init__.py'</span><span style=\"font-weight: bold\">)</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'my_pkg'</span>,\n",
|
|
||||||
" <span style=\"color: #800080; text-decoration-color: #800080; font-weight: bold\">PosixPath</span><span style=\"font-weight: bold\">(</span><span style=\"color: #008000; text-decoration-color: #008000\">'/mnt/ad_dev_conf/apps/my_repo/my_pkg/motion.py'</span><span style=\"font-weight: bold\">)</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'my_pkg'</span>,\n",
|
|
||||||
" <span style=\"color: #800080; text-decoration-color: #800080; font-weight: bold\">PosixPath</span><span style=\"font-weight: bold\">(</span><span style=\"color: #008000; text-decoration-color: #008000\">'/mnt/ad_dev_conf/apps/my_repo/my_pkg/my_sub_pkg/__init__.py'</span><span style=\"font-weight: bold\">)</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'my_pkg'</span>,\n",
|
|
||||||
" <span style=\"color: #800080; text-decoration-color: #800080; font-weight: bold\">PosixPath</span><span style=\"font-weight: bold\">(</span><span style=\"color: #008000; text-decoration-color: #008000\">'/mnt/ad_dev_conf/apps/my_repo/my_pkg/my_sub_pkg/hello.py'</span><span style=\"font-weight: bold\">)</span>: <span style=\"color: #008000; text-decoration-color: #008000\">'my_pkg'</span>\n",
|
|
||||||
"<span style=\"font-weight: bold\">}</span>\n",
|
|
||||||
"</pre>\n"
|
|
||||||
],
|
|
||||||
"text/plain": [
|
|
||||||
"\u001b[1m{\u001b[0m\n",
|
|
||||||
" \u001b[1;35mPosixPath\u001b[0m\u001b[1m(\u001b[0m\u001b[32m'/mnt/ad_dev_conf/apps/my_repo/my_pkg/__init__.py'\u001b[0m\u001b[1m)\u001b[0m: \u001b[32m'my_pkg'\u001b[0m,\n",
|
|
||||||
" \u001b[1;35mPosixPath\u001b[0m\u001b[1m(\u001b[0m\u001b[32m'/mnt/ad_dev_conf/apps/my_repo/my_pkg/motion.py'\u001b[0m\u001b[1m)\u001b[0m: \u001b[32m'my_pkg'\u001b[0m,\n",
|
|
||||||
" \u001b[1;35mPosixPath\u001b[0m\u001b[1m(\u001b[0m\u001b[32m'/mnt/ad_dev_conf/apps/my_repo/my_pkg/my_sub_pkg/__init__.py'\u001b[0m\u001b[1m)\u001b[0m: \u001b[32m'my_pkg'\u001b[0m,\n",
|
|
||||||
" \u001b[1;35mPosixPath\u001b[0m\u001b[1m(\u001b[0m\u001b[32m'/mnt/ad_dev_conf/apps/my_repo/my_pkg/my_sub_pkg/hello.py'\u001b[0m\u001b[1m)\u001b[0m: \u001b[32m'my_pkg'\u001b[0m\n",
|
|
||||||
"\u001b[1m}\u001b[0m\n"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "display_data"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"association = {}\n",
|
|
||||||
"for dir in top_packages_dirs:\n",
|
|
||||||
" associated_modules = list(dir.rglob('*.py'))\n",
|
|
||||||
" # print('\\n'.join(f' - {p.as_posix()}' for p in associated_modules))\n",
|
|
||||||
" for module_file in associated_modules:\n",
|
|
||||||
" association[module_file] = dir.stem\n",
|
|
||||||
"\n",
|
|
||||||
"print(association)"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 15,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/plain": [
|
|
||||||
"['hello_sub', 'motion']"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"execution_count": 15,
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "execute_result"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"module_name = 'my_pkg'\n",
|
|
||||||
"\n",
|
|
||||||
"with Path('apps/apps.yaml').open('r') as f:\n",
|
|
||||||
" self.app_config = yaml.load(f, Loader=yaml.SafeLoader)\n",
|
|
||||||
"\n",
|
|
||||||
"self.non_apps = []\n",
|
|
||||||
"\n",
|
|
||||||
"[\n",
|
|
||||||
" app_name\n",
|
|
||||||
" for app_name, app_cfg in self.app_config.items()\n",
|
|
||||||
" if app_cfg['module'].split('.')[0] == module_name\n",
|
|
||||||
"]"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": null,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": []
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metadata": {
|
|
||||||
"kernelspec": {
|
|
||||||
"display_name": "ad_dev",
|
|
||||||
"language": "python",
|
|
||||||
"name": "python3"
|
|
||||||
},
|
|
||||||
"language_info": {
|
|
||||||
"codemirror_mode": {
|
|
||||||
"name": "ipython",
|
|
||||||
"version": 3
|
|
||||||
},
|
|
||||||
"file_extension": ".py",
|
|
||||||
"mimetype": "text/x-python",
|
|
||||||
"name": "python",
|
|
||||||
"nbconvert_exporter": "python",
|
|
||||||
"pygments_lexer": "ipython3",
|
|
||||||
"version": "3.11.7"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"nbformat": 4,
|
|
||||||
"nbformat_minor": 2
|
|
||||||
}
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
{
|
|
||||||
"cells": [
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 1,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": [
|
|
||||||
"import importlib\n",
|
|
||||||
"from pathlib import Path"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": null,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": []
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metadata": {
|
|
||||||
"kernelspec": {
|
|
||||||
"display_name": "ad_dev",
|
|
||||||
"language": "python",
|
|
||||||
"name": "python3"
|
|
||||||
},
|
|
||||||
"language_info": {
|
|
||||||
"codemirror_mode": {
|
|
||||||
"name": "ipython",
|
|
||||||
"version": 3
|
|
||||||
},
|
|
||||||
"file_extension": ".py",
|
|
||||||
"mimetype": "text/x-python",
|
|
||||||
"name": "python",
|
|
||||||
"nbconvert_exporter": "python",
|
|
||||||
"pygments_lexer": "ipython3",
|
|
||||||
"version": "3.11.7"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"nbformat": 4,
|
|
||||||
"nbformat_minor": 2
|
|
||||||
}
|
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
appdaemon:
|
appdaemon:
|
||||||
|
# ffake_ketys: asdfasdf
|
||||||
|
# starttime: '2024-07-14 12:00:00' # YYYY-MM-DD HH:MM:SS
|
||||||
uvloop: True
|
uvloop: True
|
||||||
use_dictionary_unpacking: True
|
use_dictionary_unpacking: True
|
||||||
# import_method: expert
|
# import_method: expert
|
||||||
@@ -8,9 +10,15 @@ appdaemon:
|
|||||||
longitude: 0
|
longitude: 0
|
||||||
elevation: 30
|
elevation: 30
|
||||||
time_zone: America/Chicago
|
time_zone: America/Chicago
|
||||||
check_app_updates_profile: True
|
# check_app_updates_profile: True
|
||||||
|
module_debug:
|
||||||
|
_app_management: DEBUG
|
||||||
|
_callbacks: DEBUG
|
||||||
|
# _events: DEBUG
|
||||||
|
# _threading: DEBUG
|
||||||
|
# mqtt: DEBUG
|
||||||
plugins:
|
plugins:
|
||||||
HASS:
|
hass:
|
||||||
type: hass
|
type: hass
|
||||||
ha_url: http://192.168.1.82:8123
|
ha_url: http://192.168.1.82:8123
|
||||||
token: !secret long_lived_token
|
token: !secret long_lived_token
|
||||||
@@ -19,6 +27,7 @@ appdaemon:
|
|||||||
# namespace: mqtt
|
# namespace: mqtt
|
||||||
# client_host: zigbee.localdomain
|
# client_host: zigbee.localdomain
|
||||||
# client_user: homeassistant
|
# client_user: homeassistant
|
||||||
|
# client_id: test_dev_client
|
||||||
# client_password: !secret mqtt_password
|
# client_password: !secret mqtt_password
|
||||||
# client_topics:
|
# client_topics:
|
||||||
# - zigbee2mqtt/#
|
# - zigbee2mqtt/#
|
||||||
|
|||||||
29
apps/app1/button.py
Normal file
29
apps/app1/button.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import json
|
||||||
|
|
||||||
|
from appdaemon.adapi import ADAPI
|
||||||
|
|
||||||
|
|
||||||
|
class Button(ADAPI):
|
||||||
|
def initialize(self):
|
||||||
|
name = self.args['button']
|
||||||
|
self.handle = self.listen_event(
|
||||||
|
self.handle_button,
|
||||||
|
'MQTT_MESSAGE',
|
||||||
|
topic=f'zigbee2mqtt/{name}',
|
||||||
|
payload=self.payload_filter,
|
||||||
|
)
|
||||||
|
self.log(f"Started MQTT app in namespace '{self._namespace}'")
|
||||||
|
# raise ValueError
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def payload_filter(payload: str):
|
||||||
|
try:
|
||||||
|
return json.loads(payload)['action'] != ''
|
||||||
|
except Exception:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def handle_button(self, event_name, data, **kwargs):
|
||||||
|
data['payload'] = json.loads(data['payload'])
|
||||||
|
# if data['payload']['action'] != '':
|
||||||
|
json_str = json.dumps(data, indent=4)
|
||||||
|
self.logger.info(f'{event_name} callback with\n{json_str}\n{kwargs}')
|
||||||
@@ -1,6 +1,12 @@
|
|||||||
from appdaemon.adapi import ADAPI
|
from appdaemon.adapi import ADAPI
|
||||||
|
|
||||||
|
|
||||||
class Database:
|
class Database(ADAPI):
|
||||||
|
def initialize(self):
|
||||||
|
self.comp = DatabaseComponent(self)
|
||||||
|
self.log('Database with component')
|
||||||
|
|
||||||
|
|
||||||
|
class DatabaseComponent:
|
||||||
def __init__(self, ad: ADAPI) -> None:
|
def __init__(self, ad: ADAPI) -> None:
|
||||||
self.ad = ad
|
self.ad = ad
|
||||||
|
|||||||
3
apps/apps.yaml
Normal file
3
apps/apps.yaml
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
hello_world:
|
||||||
|
module: hello
|
||||||
|
class: HelloWorld
|
||||||
13
apps/automationlib/automationlib.py
Normal file
13
apps/automationlib/automationlib.py
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import arrow
|
||||||
|
from appdaemon.adapi import ADAPI
|
||||||
|
|
||||||
|
|
||||||
|
class AutomationLib:
|
||||||
|
"""This is the documentation for AutomationLib"""
|
||||||
|
|
||||||
|
def __init__(self, ad: ADAPI) -> None:
|
||||||
|
self.ad = ad
|
||||||
|
self.ad.log(f"AutomationLib initialised for app '{self.ad.name}'", level='DEBUG')
|
||||||
|
|
||||||
|
def dow(self):
|
||||||
|
return arrow.now().isoweekday()
|
||||||
10
apps/automationlib/automationlib.yaml
Normal file
10
apps/automationlib/automationlib.yaml
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
automationlib:
|
||||||
|
module: automationlib
|
||||||
|
global: true
|
||||||
|
|
||||||
|
my_test_app:
|
||||||
|
module: test
|
||||||
|
class: Test
|
||||||
|
log_level: DEBUG
|
||||||
|
dependencies:
|
||||||
|
- automationlib
|
||||||
1
apps/automationlib/requirements.txt
Normal file
1
apps/automationlib/requirements.txt
Normal file
@@ -0,0 +1 @@
|
|||||||
|
arrow
|
||||||
14
apps/automationlib/test.py
Normal file
14
apps/automationlib/test.py
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
from appdaemon.plugins.hass.hassapi import Hass
|
||||||
|
from automationlib import AutomationLib
|
||||||
|
|
||||||
|
|
||||||
|
class Test(Hass):
|
||||||
|
"""This is the documentation for Test"""
|
||||||
|
|
||||||
|
def initialize(self):
|
||||||
|
"""."""
|
||||||
|
|
||||||
|
self.lib = AutomationLib(self)
|
||||||
|
self.log('-' * 72)
|
||||||
|
self.log(f'Day of the week: {self.lib.dow()}')
|
||||||
|
self.log('-' * 72)
|
||||||
6
apps/button.yaml
Normal file
6
apps/button.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
button_test:
|
||||||
|
module: app1.button
|
||||||
|
class: Button
|
||||||
|
button: Living Room Button
|
||||||
|
dependencies:
|
||||||
|
- hello_world
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
App1:
|
App1:
|
||||||
module: aadfasdf
|
module: app1.database
|
||||||
class: Database
|
class: Database
|
||||||
other_kwargs: value
|
other_kwargs: value
|
||||||
dependencies: Sound
|
dependencies: Sound
|
||||||
@@ -17,3 +17,9 @@ App2:
|
|||||||
- App1
|
- App1
|
||||||
- App1Foo
|
- App1Foo
|
||||||
- globals
|
- globals
|
||||||
|
|
||||||
|
App3:
|
||||||
|
module: app2.database
|
||||||
|
class: OtherDatabaseApp
|
||||||
|
dependencies:
|
||||||
|
- App2
|
||||||
|
|||||||
0
apps/family/__init__.py
Normal file
0
apps/family/__init__.py
Normal file
10
apps/family/child.py
Normal file
10
apps/family/child.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from appdaemon.adapi import ADAPI
|
||||||
|
|
||||||
|
from .utils import UTILS_VALS
|
||||||
|
|
||||||
|
|
||||||
|
raise ImportError('Fake import error')
|
||||||
|
|
||||||
|
class Child(ADAPI):
|
||||||
|
def initialize(self):
|
||||||
|
self.log(f'Initialized child with: {UTILS_VALS}')
|
||||||
7
apps/family/family.yaml
Normal file
7
apps/family/family.yaml
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
parent:
|
||||||
|
module: family.parent
|
||||||
|
class: Parent
|
||||||
|
|
||||||
|
child:
|
||||||
|
module: family.child
|
||||||
|
class: Child
|
||||||
8
apps/family/parent.py
Normal file
8
apps/family/parent.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from appdaemon.adapi import ADAPI
|
||||||
|
|
||||||
|
from .utils import UTILS_VALS
|
||||||
|
|
||||||
|
|
||||||
|
class Parent(ADAPI):
|
||||||
|
def initialize(self):
|
||||||
|
self.log(f'Initialized parent with: {UTILS_VALS}')
|
||||||
0
apps/family/relatives/__init__.py
Normal file
0
apps/family/relatives/__init__.py
Normal file
1
apps/family/relatives/city.py
Normal file
1
apps/family/relatives/city.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
OTHER_CITY = 'NYC again'
|
||||||
3
apps/family/utils.py
Normal file
3
apps/family/utils.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
from .relatives.city import OTHER_CITY
|
||||||
|
|
||||||
|
UTILS_VALS = 100, 600
|
||||||
@@ -5,4 +5,4 @@ from appdaemon.adapi import ADAPI
|
|||||||
|
|
||||||
class HelloWorld(ADAPI):
|
class HelloWorld(ADAPI):
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
self.log(f'Initialized app from {Path(__file__).relative_to(self.AD.app_dir.parent)}')
|
self.log(f'Initialized app from {Path(__file__).relative_to(Path(self.AD.app_dir).parent)}')
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
rich
|
rich
|
||||||
anyio
|
anyio
|
||||||
pydantic
|
pydantic
|
||||||
|
# arrow
|
||||||
@@ -4,4 +4,8 @@ test_notifier:
|
|||||||
module: variant
|
module: variant
|
||||||
class: Variant
|
class: Variant
|
||||||
log_level: DEBUG
|
log_level: DEBUG
|
||||||
# dependencies: companion_app
|
# dependencies: companion_app
|
||||||
|
|
||||||
|
TestApp:
|
||||||
|
module: test_notify
|
||||||
|
class: TestNotifier
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ from appdaemon.adapi import ADAPI
|
|||||||
|
|
||||||
class Variant(TestNotifier, ADAPI):
|
class Variant(TestNotifier, ADAPI):
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
self.adapi = self.get_ad_api()
|
super().initialize()
|
||||||
self.log = self.adapi.log
|
|
||||||
self.log('Variant of the TestNotifier')
|
self.log('Variant of the TestNotifier')
|
||||||
# self.log(' VARIANT '.center(50, '*'))
|
# self.log(' VARIANT '.center(50, '*'))
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ services:
|
|||||||
appdaemon:
|
appdaemon:
|
||||||
container_name: appdaemon_dev
|
container_name: appdaemon_dev
|
||||||
image: appdaemon:local-dev
|
image: appdaemon:local-dev
|
||||||
|
# image: acockburn/appdaemon:dev
|
||||||
volumes:
|
volumes:
|
||||||
- /etc/localtime:/etc/localtime:ro
|
- /etc/localtime:/etc/localtime:ro
|
||||||
- /etc/timezone:/etc/timezone:ro
|
- /etc/timezone:/etc/timezone:ro
|
||||||
|
|||||||
144
modules.ipynb
144
modules.ipynb
@@ -1,144 +0,0 @@
|
|||||||
{
|
|
||||||
"cells": [
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 1,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": [
|
|
||||||
"import importlib\n",
|
|
||||||
"\n",
|
|
||||||
"import pathlib\n",
|
|
||||||
"from types import ModuleType\n",
|
|
||||||
"from typing import List\n",
|
|
||||||
"from pathlib import Path\n",
|
|
||||||
"\n",
|
|
||||||
"from IPython.lib import deepreload"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 2,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"name": "stdout",
|
|
||||||
"output_type": "stream",
|
|
||||||
"text": [
|
|
||||||
"=============== Importing my_pkg ===============\n",
|
|
||||||
"-------------- Importing motion.py ---------------\n",
|
|
||||||
"===================== Done =====================\n"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"import sys\n",
|
|
||||||
"\n",
|
|
||||||
"sys.path.insert(0, Path('apps/my_repo').resolve().as_posix())\n",
|
|
||||||
"\n",
|
|
||||||
"import my_pkg"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 3,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"data": {
|
|
||||||
"text/plain": [
|
|
||||||
"['importlib._abc',\n",
|
|
||||||
" 'importlib._bootstrap',\n",
|
|
||||||
" 'importlib._bootstrap_external',\n",
|
|
||||||
" 'importlib._imp',\n",
|
|
||||||
" 'importlib.abc',\n",
|
|
||||||
" 'importlib.machinery',\n",
|
|
||||||
" 'importlib.metadata',\n",
|
|
||||||
" 'importlib.resources',\n",
|
|
||||||
" 'importlib.sys',\n",
|
|
||||||
" 'importlib.util',\n",
|
|
||||||
" 'importlib.warnings']"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"execution_count": 3,
|
|
||||||
"metadata": {},
|
|
||||||
"output_type": "execute_result"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"def recursive_modules(module_obj: ModuleType) -> List[str]:\n",
|
|
||||||
" sub_modules = [\n",
|
|
||||||
" f'{module_obj.__name__}.{name}' for name in dir(module_obj)\n",
|
|
||||||
" if isinstance(getattr(module_obj, name), ModuleType)\n",
|
|
||||||
" ]\n",
|
|
||||||
" return sub_modules\n",
|
|
||||||
"\n",
|
|
||||||
"recursive_modules(importlib)"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": 10,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [
|
|
||||||
{
|
|
||||||
"name": "stdout",
|
|
||||||
"output_type": "stream",
|
|
||||||
"text": [
|
|
||||||
"[<module 'my_pkg' from '/home/john/conf/apps/my_repo/my_pkg/__init__.py'>, <module 'my_pkg.motion' from '/home/john/conf/apps/my_repo/my_pkg/motion.py'>, <module 'my_pkg.my_sub_pkg' from '/home/john/conf/apps/my_repo/my_pkg/my_sub_pkg/__init__.py'>, <module 'my_pkg.my_sub_pkg.hello' from '/home/john/conf/apps/my_repo/my_pkg/my_sub_pkg/hello.py'>]\n",
|
|
||||||
"=============== Importing my_pkg ===============\n",
|
|
||||||
"===================== Done =====================\n",
|
|
||||||
"-------------- Importing motion.py ---------------\n"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"source": [
|
|
||||||
"def recursive_modules(module: ModuleType):\n",
|
|
||||||
" for name in dir(module):\n",
|
|
||||||
" sub_mod = getattr(module, name)\n",
|
|
||||||
" if isinstance(sub_mod, ModuleType) and sub_mod.__name__.startswith(module.__name__):\n",
|
|
||||||
" yield from recursive_modules(sub_mod)\n",
|
|
||||||
" yield module\n",
|
|
||||||
"\n",
|
|
||||||
"\n",
|
|
||||||
"def reload_package(pkg: ModuleType):\n",
|
|
||||||
" modules = sorted(recursive_modules(pkg), key=lambda m: m.__name__)\n",
|
|
||||||
" for mod in modules:\n",
|
|
||||||
" importlib.reload(mod)\n",
|
|
||||||
"\n",
|
|
||||||
"\n",
|
|
||||||
"reload_package(my_pkg)"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"cell_type": "code",
|
|
||||||
"execution_count": null,
|
|
||||||
"metadata": {},
|
|
||||||
"outputs": [],
|
|
||||||
"source": [
|
|
||||||
"importlib.import_module(name)"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"metadata": {
|
|
||||||
"kernelspec": {
|
|
||||||
"display_name": "appdaemon",
|
|
||||||
"language": "python",
|
|
||||||
"name": "python3"
|
|
||||||
},
|
|
||||||
"language_info": {
|
|
||||||
"codemirror_mode": {
|
|
||||||
"name": "ipython",
|
|
||||||
"version": 3
|
|
||||||
},
|
|
||||||
"file_extension": ".py",
|
|
||||||
"mimetype": "text/x-python",
|
|
||||||
"name": "python",
|
|
||||||
"nbconvert_exporter": "python",
|
|
||||||
"pygments_lexer": "ipython3",
|
|
||||||
"version": "3.11.8"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"nbformat": 4,
|
|
||||||
"nbformat_minor": 2
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user