mirror of
https://github.com/serega404/VodokanalBot.git
synced 2026-05-30 12:10:01 +03:00
Добавил поддержку MeshCore через HA
This commit is contained in:
@@ -11,6 +11,13 @@ env:
|
|||||||
jobs:
|
jobs:
|
||||||
build-and-push-image:
|
build-and-push-image:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
image:
|
||||||
|
- suffix: ''
|
||||||
|
start_file: start_telegram.py
|
||||||
|
- suffix: '-meshcore-ha'
|
||||||
|
start_file: start_meshcore_ha.py
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
packages: write
|
packages: write
|
||||||
@@ -30,12 +37,14 @@ jobs:
|
|||||||
id: meta
|
id: meta
|
||||||
uses: docker/metadata-action@98669ae865ea3cffbcbaa878cf57c20bbf1c6c38
|
uses: docker/metadata-action@98669ae865ea3cffbcbaa878cf57c20bbf1c6c38
|
||||||
with:
|
with:
|
||||||
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}${{ matrix.image.suffix }}
|
||||||
|
|
||||||
- name: Build and push Docker image
|
- name: Build and push Docker image
|
||||||
uses: docker/build-push-action@ad44023a93711e3deb337508980b4b5e9bcdc5dc
|
uses: docker/build-push-action@ad44023a93711e3deb337508980b4b5e9bcdc5dc
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
|
build-args: |
|
||||||
|
START_FILE=${{ matrix.image.start_file }}
|
||||||
push: true
|
push: true
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
+3
-2
@@ -1,8 +1,9 @@
|
|||||||
FROM python:3.11.2-alpine3.17
|
FROM python:3.12-alpine
|
||||||
|
|
||||||
LABEL Maintainer="serega404"
|
LABEL Maintainer="serega404"
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
ARG START_FILE=start_telegram.py
|
||||||
|
|
||||||
COPY requirements.txt requirements.txt
|
COPY requirements.txt requirements.txt
|
||||||
RUN pip3 install -r requirements.txt
|
RUN pip3 install -r requirements.txt
|
||||||
@@ -11,7 +12,7 @@ RUN pip3 install -r requirements.txt
|
|||||||
COPY crontab /tmp/crontab
|
COPY crontab /tmp/crontab
|
||||||
RUN cat /tmp/crontab > /etc/crontabs/root
|
RUN cat /tmp/crontab > /etc/crontabs/root
|
||||||
|
|
||||||
COPY start_telegram.py start_telegram.py
|
COPY ${START_FILE} start.py
|
||||||
COPY parser.py parser.py
|
COPY parser.py parser.py
|
||||||
|
|
||||||
# run crond as main process of container
|
# run crond as main process of container
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
[](https://github.com/serega404/VodokanalBot)
|
[](https://github.com/serega404/VodokanalBot)
|
||||||
|
|
||||||
## Запуск в Docker
|
## Запуск TG интеграции в Docker
|
||||||
|
|
||||||
``` Docker
|
``` Docker
|
||||||
docker volume create vodokanal_bot_data
|
docker volume create vodokanal_bot_data
|
||||||
@@ -15,7 +15,7 @@ docker run -d --name VodokanalBot \
|
|||||||
ghcr.io/serega404/vodokanalbot:main
|
ghcr.io/serega404/vodokanalbot:main
|
||||||
```
|
```
|
||||||
|
|
||||||
## Запуск в Docker Compose
|
### Запуск в Docker Compose
|
||||||
|
|
||||||
Укажи `TELEGRAM_TOKEN` и `TELEGRAM_CHANNEL` в [`docker-compose.yml`](./docker-compose.yml), затем запусти:
|
Укажи `TELEGRAM_TOKEN` и `TELEGRAM_CHANNEL` в [`docker-compose.yml`](./docker-compose.yml), затем запусти:
|
||||||
|
|
||||||
@@ -23,6 +23,33 @@ docker run -d --name VodokanalBot \
|
|||||||
docker compose up -d --build
|
docker compose up -d --build
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Запуск с MeshCore + Home Assistant
|
||||||
|
|
||||||
|
Вариант для MeshCore отправляет новые сообщения в webhook Home Assistant, а Home Assistant пересылает их в канал MeshCore.
|
||||||
|
|
||||||
|
1. Добавь в Home Assistant автоматизацию [`Webhook to MeshCore channel`](https://github.com/serega404/HA-MeshCore#webhook-to-meshcore-channel).
|
||||||
|
2. Замени `CHANGE_ME_RANDOM_WEBHOOK_ID` в автоматизации на случайную строку.
|
||||||
|
|
||||||
|
``` Docker
|
||||||
|
docker volume create vodokanal_bot_data
|
||||||
|
docker run -d --name VodokanalBot \
|
||||||
|
--restart=always \
|
||||||
|
-v vodokanal_bot_data:/app/data \
|
||||||
|
-e TZ='Europe/Moscow' \
|
||||||
|
-e HOME_ASSISTANT_WEBHOOK_ID='CHANGE_ME_RANDOM_WEBHOOK_ID' \
|
||||||
|
-e HOME_ASSISTANT_URL='http://homeassistant.local:8123' \
|
||||||
|
-e HOME_ASSISTANT_WEBHOOK_CHANNEL='0' \
|
||||||
|
ghcr.io/serega404/vodokanalbot-meshcore-ha:main
|
||||||
|
```
|
||||||
|
|
||||||
|
### Запуск в Docker Compose
|
||||||
|
|
||||||
|
Укажи `HOME_ASSISTANT_WEBHOOK_ID` и `HOME_ASSISTANT_WEBHOOK_CHANNEL` в [`docker-compose.ha.yml`](./docker-compose.ha.yml), затем запусти:
|
||||||
|
|
||||||
|
``` Docker
|
||||||
|
docker compose up -d --build -f docker-compose.ha.yml
|
||||||
|
```
|
||||||
|
|
||||||
## Интеграции
|
## Интеграции
|
||||||
|
|
||||||
Общая логика парсинга, работы с `data/db.json` и поиска новых сообщений вынесена в [`parser.py`](./parser.py).
|
Общая логика парсинга, работы с `data/db.json` и поиска новых сообщений вынесена в [`parser.py`](./parser.py).
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
@reboot cd /app && python3 /app/start_telegram.py
|
@reboot cd /app && python3 /app/start.py
|
||||||
*/10 * * * * cd /app && python3 /app/start_telegram.py
|
*/10 * * * * cd /app && python3 /app/start.py
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
services:
|
||||||
|
vodokanalbot-ha:
|
||||||
|
image: ghcr.io/serega404/vodokanalbot-meshcore-ha:main
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
args:
|
||||||
|
START_FILE: start_meshcore_ha.py
|
||||||
|
container_name: VodokanalBotMeshCoreHA
|
||||||
|
restart: always
|
||||||
|
environment:
|
||||||
|
TZ: Europe/Moscow
|
||||||
|
HOME_ASSISTANT_URL: https://your-home-assistant-url
|
||||||
|
HOME_ASSISTANT_WEBHOOK_ID: CHANGE_ME_RANDOM_WEBHOOK_ID
|
||||||
|
HOME_ASSISTANT_WEBHOOK_CHANNEL: "0"
|
||||||
|
# VODOKANAL_URL: http://www.tgnvoda.ru/avarii.php
|
||||||
|
# PROXY_URL: socks5h://user:password@proxy-host:1080
|
||||||
|
# PROXY_URL: http://user:password@proxy-host:3128
|
||||||
|
volumes:
|
||||||
|
- vodokanal_bot_data:/app/data
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
vodokanal_bot_data:
|
||||||
@@ -1,12 +1,17 @@
|
|||||||
services:
|
services:
|
||||||
vodokanalbot:
|
vodokanalbot:
|
||||||
image: ghcr.io/serega404/vodokanalbot:main
|
image: ghcr.io/serega404/vodokanalbot:main
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
args:
|
||||||
|
START_FILE: start_telegram.py
|
||||||
container_name: VodokanalBot
|
container_name: VodokanalBot
|
||||||
restart: always
|
restart: always
|
||||||
environment:
|
environment:
|
||||||
TZ: Europe/Moscow
|
TZ: Europe/Moscow
|
||||||
TELEGRAM_TOKEN: TOKEN
|
TELEGRAM_TOKEN: TOKEN
|
||||||
TELEGRAM_CHANNEL: CHAT_ID
|
TELEGRAM_CHANNEL: CHAT_ID
|
||||||
|
# VODOKANAL_URL: http://www.tgnvoda.ru/avarii.php
|
||||||
# PROXY_URL: socks5h://user:password@proxy-host:1080
|
# PROXY_URL: socks5h://user:password@proxy-host:1080
|
||||||
# PROXY_URL: http://user:password@proxy-host:3128
|
# PROXY_URL: http://user:password@proxy-host:3128
|
||||||
volumes:
|
volumes:
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ from bs4 import BeautifulSoup
|
|||||||
|
|
||||||
|
|
||||||
DEFAULT_DB_PATH = "data/db.json"
|
DEFAULT_DB_PATH = "data/db.json"
|
||||||
|
DEFAULT_VODOKANAL_URL = "http://www.tgnvoda.ru/avarii.php"
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
@dataclass(frozen=True)
|
||||||
class Post:
|
class Post:
|
||||||
@@ -31,7 +31,6 @@ def create_session(proxy_url=""):
|
|||||||
|
|
||||||
return session
|
return session
|
||||||
|
|
||||||
|
|
||||||
def load_database(path=DEFAULT_DB_PATH):
|
def load_database(path=DEFAULT_DB_PATH):
|
||||||
if not os.path.isfile(path):
|
if not os.path.isfile(path):
|
||||||
print("Database not loaded")
|
print("Database not loaded")
|
||||||
@@ -89,9 +88,9 @@ def get_new_posts(posts, database):
|
|||||||
return [post for post in posts if post.key not in database_keys]
|
return [post for post in posts if post.key not in database_keys]
|
||||||
|
|
||||||
|
|
||||||
def publish_new_posts(send_message, session, url, db_path=DEFAULT_DB_PATH):
|
def publish_new_posts(send_message, session, db_path=DEFAULT_DB_PATH):
|
||||||
database = load_database(db_path)
|
database = load_database(db_path)
|
||||||
posts = fetch_posts(session, url)
|
posts = fetch_posts(session, DEFAULT_VODOKANAL_URL)
|
||||||
|
|
||||||
if not posts:
|
if not posts:
|
||||||
print("No posts")
|
print("No posts")
|
||||||
|
|||||||
@@ -0,0 +1,56 @@
|
|||||||
|
import os
|
||||||
|
|
||||||
|
from parser import create_session, publish_new_posts
|
||||||
|
|
||||||
|
|
||||||
|
PROXY_URL = os.environ.get('PROXY_URL', '')
|
||||||
|
HOME_ASSISTANT_URL = os.environ.get('HOME_ASSISTANT_URL', '')
|
||||||
|
HOME_ASSISTANT_WEBHOOK_ID = os.environ.get('HOME_ASSISTANT_WEBHOOK_ID', '')
|
||||||
|
HOME_ASSISTANT_WEBHOOK_CHANNEL = os.environ.get('HOME_ASSISTANT_WEBHOOK_CHANNEL', '0')
|
||||||
|
|
||||||
|
|
||||||
|
def create_webhook_url():
|
||||||
|
return (
|
||||||
|
HOME_ASSISTANT_URL.rstrip('/')
|
||||||
|
+ "/api/webhook/"
|
||||||
|
+ HOME_ASSISTANT_WEBHOOK_ID
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def send_webhook_message(session, message):
|
||||||
|
req = session.get(
|
||||||
|
create_webhook_url(),
|
||||||
|
params={
|
||||||
|
'channel': HOME_ASSISTANT_WEBHOOK_CHANNEL,
|
||||||
|
'message': message,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not 200 <= req.status_code < 300:
|
||||||
|
print("Home Assistant webhook request error: " + str(req.status_code))
|
||||||
|
exit()
|
||||||
|
else:
|
||||||
|
print("Home Assistant webhook message sent")
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if HOME_ASSISTANT_URL == '':
|
||||||
|
print("Home Assistant URL is not set")
|
||||||
|
exit()
|
||||||
|
|
||||||
|
if HOME_ASSISTANT_WEBHOOK_ID == '':
|
||||||
|
print("Home Assistant webhook id is not set")
|
||||||
|
exit()
|
||||||
|
|
||||||
|
session = create_session(PROXY_URL)
|
||||||
|
try:
|
||||||
|
publish_new_posts(
|
||||||
|
send_message=lambda message: send_webhook_message(session, message),
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
except RuntimeError as error:
|
||||||
|
print(error)
|
||||||
|
exit()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -3,7 +3,6 @@ import os
|
|||||||
from parser import create_session, publish_new_posts
|
from parser import create_session, publish_new_posts
|
||||||
|
|
||||||
|
|
||||||
URL = os.environ.get('VODOKANAL_URL', 'http://www.tgnvoda.ru/avarii.php')
|
|
||||||
SEND_SILENT = os.environ.get('SEND_SILENT', False)
|
SEND_SILENT = os.environ.get('SEND_SILENT', False)
|
||||||
TELEGRAM_TOKEN = os.environ.get('TELEGRAM_TOKEN', '')
|
TELEGRAM_TOKEN = os.environ.get('TELEGRAM_TOKEN', '')
|
||||||
TELEGRAM_CHANNEL = os.environ.get('TELEGRAM_CHANNEL', '')
|
TELEGRAM_CHANNEL = os.environ.get('TELEGRAM_CHANNEL', '')
|
||||||
@@ -40,7 +39,6 @@ def main():
|
|||||||
publish_new_posts(
|
publish_new_posts(
|
||||||
send_message=lambda message: send_telegram_message(session, message),
|
send_message=lambda message: send_telegram_message(session, message),
|
||||||
session=session,
|
session=session,
|
||||||
url=URL,
|
|
||||||
)
|
)
|
||||||
except RuntimeError as error:
|
except RuntimeError as error:
|
||||||
print(error)
|
print(error)
|
||||||
|
|||||||
Reference in New Issue
Block a user