Added ability to send notifications in Matrix

This commit is contained in:
Anry Das 2025-11-29 11:51:12 +02:00
parent 7060f34d85
commit c2b024e5df
3 changed files with 130 additions and 48 deletions

168
get_eo.py
View File

@ -19,10 +19,10 @@ STATUS_ERROR_SEND_TO_TG = 20
STATUS_ERROR_GPV_RESPONSE = 10
STATUS_ERROR_PO_RESPONSE = 30
STATUS_EMPTY_CONFIG = 5
STATUS_ERROR_SEND_TO_MX = 40
SCRIPT_PATH = os.path.dirname(__file__)
#CONFIG_FILE_NAME = SCRIPT_PATH + '/get_eo_config.yaml'
CONFIG_FILE_NAME = SCRIPT_PATH + '/eo_config.json'
CONFIG_FILE_NAME = SCRIPT_PATH + '/get_eo_config.yaml'
LAST_DATAS_FILE_NAME_GPV = SCRIPT_PATH + '/get_eo.last_info'
LAST_DATAS_FILE_NAME_PO = SCRIPT_PATH + '/get_po.last_info'
LAST_DATAS_FORMAT_GPV = '{array} {date} {day}'
@ -30,9 +30,14 @@ LAST_DATAS_FORMAT_PO = '{placed} {star_dt} {start_tm} {stop_dt} {stop_tm} {curre
IS_DEBUG = False
MENTIONS = ""
USE_MATRIX = False
MX_SERVER = ""
MX_ROOM = ""
MX_TOKEN = ""
def read_config():
global MENTIONS
global MENTIONS, USE_MATRIX, MX_SERVER, MX_ROOM, MX_TOKEN
j = read_cfg(CONFIG_FILE_NAME)
tg_token = j['token']
tg_chat = j['chat_id']
@ -43,17 +48,53 @@ def read_config():
house = j['house']
building_part_number = j['building_part_number']
apartment = j['apartment']
for m in j['mentions']:
MENTIONS += m + '\n'
USE_MATRIX = j['use_matrix'].lower() == 'true'
MX_SERVER = j['mx_server']
MX_ROOM = j['mx_room_id']
MX_TOKEN = j['mx_access_token']
# i = 1
# for m in j['mentions']:
# MENTIONS += f'<a href="{m}">user {i}</a>' + '\n'
# i += 1
return tg_token, tg_chat, oe_account_number, is_debug, settlement, street, house, building_part_number, apartment
def send_message_to_tg(msg, token, chat):
url = f"https://api.telegram.org/bot{token}/sendMessage?chat_id={chat}&parse_mode=html&text={msg}"
return requests.get(url)
# return requests.post(
# url='https://api.telegram.org/bot{0}/{1}'.format(token, 'sendMessage'),
# data={'chat_id': {chat}, 'text': {msg}, 'parse_mode': 'html'}
# )
url = f"https://api.telegram.org/bot{token}/sendMessage?chat_id={chat}&parse_mode=html&text={msg}"
return requests.get(url)
# return requests.post(
# url='https://api.telegram.org/bot{0}/{1}'.format(token, 'sendMessage'),
# data={'chat_id': {chat}, 'text': {msg}, 'parse_mode': 'html'}
# )
def send_matrix_notification(message: str) -> bool:
"""Send notification to Matrix"""
if not USE_MATRIX:
return False
try:
url = f"{MX_SERVER}/_matrix/client/r0/rooms/{MX_ROOM}/send/m.room.message"
headers = {
'Authorization': f'Bearer {MX_TOKEN}',
'Content-Type': 'application/json'
}
payload = {
'msgtype': 'm.text',
'body': message,
'format': 'org.matrix.custom.html',
'formatted_body': message.replace('\n', '<br/>')
}
response = requests.post(url, headers=headers, json=payload)
return response.status_code == 200
except Exception as e:
# logger.error(f"Error sending Matrix notification: {e}")
return False
def get_GPV_response_from_oe(account):
@ -80,7 +121,8 @@ def get_GPV_response_from_oe(account):
urllib3.disable_warnings()
return requests.post(url, headers=headers, data=data, verify=False)
def get_PO_response(settlement, street, house, building_part_number='', apartment=''): # Power Outage
def get_PO_response(settlement, street, house, building_part_number='', apartment=''): # Power Outage
utf8 = ''
url = 'https://oe.if.ua/uk/shutdowns_table'
params = {
@ -89,7 +131,7 @@ def get_PO_response(settlement, street, house, building_part_number='', apartmen
'street': street,
'house_number': house,
'building_part_number': building_part_number,
'apartment' : apartment,
'apartment': apartment,
'commit': 'Пошук'
}
headers = {
@ -99,17 +141,19 @@ def get_PO_response(settlement, street, house, building_part_number='', apartmen
urllib3.disable_warnings()
return requests.get(url=url, headers=headers, params=params)
def has_numbers(input):
return any(char.isdigit() for char in str(input))
def get_utf_chars(number, is_time=False):
digit_chars = ['\U00000030\U000020E3', '\U00000031\U000020E3', '\U00000032\U000020E3', '\U00000033\U000020E3',
'\U00000034\U000020E3', '\U00000035\U000020E3', '\U00000036\U000020E3', '\U00000037\U000020E3',
'\U00000038\U000020E3', '\U00000039\U000020E3']
res = ''
if is_time:
h = int(number)
m = int((number * 60) % 60)
h = number.split(':')[0]
m = number.split(':')[1]
if m != 0:
value = str(h) + ':' + str(m)
else:
@ -123,28 +167,33 @@ def get_utf_chars(number, is_time=False):
res += c
return res
def get_digit_message(arr, is_time=False):
first = 0
if len(arr)> 0 and arr[0] > 0:
if len(arr) > 0:
first = arr[0]
last = 0
if len(arr)> 1 and arr[0] > 0:
if len(arr) > 1:
last = arr[1]
msg = 'з {num0} до {num1}'
return msg.format(num0=get_utf_chars(first, is_time), num1=get_utf_chars(last, is_time))
def save_last_datas_GPV(hours_on, hours_off, date, day):
with open(LAST_DATAS_FILE_NAME_GPV, 'w') as f:
f.write(LAST_DATAS_FORMAT_GPV.format(array=str(hours_on)+str(hours_off), date=date, day=day))
f.write(LAST_DATAS_FORMAT_GPV.format(array=str(hours_on) + str(hours_off), date=date, day=day))
def save_last_datas_PO(placed='', star_dt='', start_tm='', stop_dt='', stop_tm=''):
date = dt.today().strftime('%Y-%m-%d')
with open(LAST_DATAS_FILE_NAME_PO, 'w') as f:
f.write(LAST_DATAS_FORMAT_PO.format(placed=str(placed), star_dt=str(star_dt), start_tm=str(start_tm), stop_dt=str(stop_dt), stop_tm=str(stop_tm),
f.write(LAST_DATAS_FORMAT_PO.format(placed=str(placed), star_dt=str(star_dt), start_tm=str(start_tm),
stop_dt=str(stop_dt), stop_tm=str(stop_tm),
current_date=date))
def load_last_datas(file_name):
if os.path.isfile(file_name):
with open(file_name, 'r') as f:
@ -152,16 +201,20 @@ def load_last_datas(file_name):
else:
return ''
def is_old_and_new_datas_equals_GPV(hours_on, hours_off, date, day):
loaded = load_last_datas(LAST_DATAS_FILE_NAME_GPV)
return LAST_DATAS_FORMAT_GPV.format(array=str(hours_on)+str(hours_off), date=date, day=day) == loaded
return LAST_DATAS_FORMAT_GPV.format(array=str(hours_on) + str(hours_off), date=date, day=day) == loaded
def is_old_and_new_datas_equals_PO(placed='', star_dt='', start_tm='', stop_dt='', stop_tm=''):
date = dt.today().strftime('%Y-%m-%d')
loaded = load_last_datas(LAST_DATAS_FILE_NAME_PO)
return LAST_DATAS_FORMAT_PO.format(placed=str(placed), star_dt=str(star_dt), start_tm=str(start_tm), stop_dt=str(stop_dt),
return LAST_DATAS_FORMAT_PO.format(placed=str(placed), star_dt=str(star_dt), start_tm=str(start_tm),
stop_dt=str(stop_dt),
stop_tm=str(stop_tm), current_date=date) == loaded
def get_GPV_message(approved_from, event_date, hours_off, hours_on, outage_queue):
queue = get_utf_chars(outage_queue)
message = f'\U000026A1 <b>ГПВ на {event_date} для {queue} черги</b>'
@ -171,55 +224,66 @@ def get_GPV_message(approved_from, event_date, hours_off, hours_on, outage_queue
if len(hours_off) > 0:
message += '\nВимкнення:'
for i in range(len(hours_off)):
arr = [int(hours_off[i].split(':')[0]), int(hours_on[i].split(':')[0])]
arr = [hours_off[i], hours_on[i]]
msg = get_digit_message(arr, True)
message += '\n' + msg
message += (f'\n{MENTIONS}')
# message += (f'\n{MENTIONS}')
else:
message += '\nВимкнень немає \U0001F44C \U0001F483'
message += (f'\n<b><i>{approved_from}</i></b>')
return message, need_to_send
def get_address(city, street, house):
return f'{city}, {street}, {house}'
def is_outage_date_before_or_same(stop_date):
parsed_date = dt.strptime(stop_date, "%Y.%m.%d").date()
current_date = dt.now().date()
return current_date <= parsed_date
def get_message_with_PO(root, html): #city, street, house, placement_date, PO_type, reason, start_date, stop_date, start_time, stop_time
def get_message_with_PO(root,
html): # city, street, house, placement_date, PO_type, reason, start_date, stop_date, start_time, stop_time
if "<div class='shutdowns'>" in html:
city = root.xpath("//div[@class='table-row flex']/div[@class='city']/text()")[0]
street = root.xpath("//div[@class='table-row flex']/div[@class='street']/text()")[0]
house = root.xpath("//div[@class='table-row flex']/div[@class='house_number']/text()")[0]
PO_type = root.xpath("//div[@class='table-row flex']/div[@class='type-shutdown']/text()")[0]
reason = root.xpath("//div[@class='table-row flex']/div[@class='reason-shutdown reason-text-container']/div[@class='reason-text-hide']/text()")[0]
placement_date = root.xpath("//div[@class='table-row flex']/div[@class='placement_date']/div[@class='date']/text()")[0]
start_date = root.xpath("//div[@class='table-row flex']/div[@class='shutdown_date']/div[@class='date']/text()")[0]
reason = root.xpath(
"//div[@class='table-row flex']/div[@class='reason-shutdown reason-text-container']/div[@class='reason-text-hide']/text()")[
0]
placement_date = \
root.xpath("//div[@class='table-row flex']/div[@class='placement_date']/div[@class='date']/text()")[0]
start_date = root.xpath("//div[@class='table-row flex']/div[@class='shutdown_date']/div[@class='date']/text()")[
0]
stop_date = root.xpath("//div[@class='table-row flex']/div[@class='turn_on_date']/div[@class='date']/text()")[0]
start_time = root.xpath("//div[@class='table-row flex']/div[@class='shutdown_date']/div[@class='time']/text()")[0]
start_time = root.xpath("//div[@class='table-row flex']/div[@class='shutdown_date']/div[@class='time']/text()")[
0]
stop_time = root.xpath("//div[@class='table-row flex']/div[@class='turn_on_date']/div[@class='time']/text()")[0]
address = get_address(city, street, house)
message = (f'\U000026A0 <b>Увага!</b>\nНа {start_date} за адресою {address} заплановано <b>{PO_type}</b> відключення (заявка від {placement_date}).\n'
f'Причина відключення: {reason}\n'
f'Початок вимкнення <b>{start_date} об {start_time}</b>\n'
f'Кінець вимкнення <b>{stop_date} об {stop_time}</b>\n'
f'{MENTIONS}')
message = (
f'\U000026A0 <b>Увага!</b>\nНа {start_date} за адресою {address} заплановано <b>{PO_type}</b> відключення (заявка від {placement_date}).\n'
f'Причина відключення: {reason}\n'
f'Початок вимкнення <b>{start_date} об {start_time}</b>\n'
f'Кінець вимкнення <b>{stop_date} об {stop_time}</b>\n'
f'{MENTIONS}')
need_to_send = (not is_old_and_new_datas_equals_PO(placed=placement_date, star_dt=start_date,
start_tm=start_time, stop_dt=stop_date, stop_tm=stop_time))
start_tm=start_time, stop_dt=stop_date, stop_tm=stop_time))
if is_outage_date_before_or_same(stop_date):
need_to_send = need_to_send and True
else:
message = '\nПланових вимкнень немає \U0001F44C \U0001F483'
need_to_send = not is_old_and_new_datas_equals_PO(placed=placement_date, star_dt=start_date,
start_tm=start_time, stop_dt=stop_date, stop_tm=stop_time)
start_tm=start_time, stop_dt=stop_date, stop_tm=stop_time)
if need_to_send:
save_last_datas_PO(placed=placement_date, star_dt=start_date, start_tm=start_time, stop_dt=stop_date, stop_tm=stop_time)
save_last_datas_PO(placed=placement_date, star_dt=start_date, start_tm=start_time, stop_dt=stop_date,
stop_tm=stop_time)
else:
message = '\nПланових вимкнень немає \U0001F44C \U0001F483'
need_to_send = not is_old_and_new_datas_equals_PO()
@ -228,6 +292,7 @@ def get_message_with_PO(root, html): #city, street, house, placement_date, PO_ty
return message, need_to_send
def process_GPV(oe_account_number, tg_chat, tg_token):
global IS_DEBUG
if not IS_DEBUG:
@ -249,7 +314,7 @@ def process_GPV(oe_account_number, tg_chat, tg_token):
datas = ''
if 'queues' in str(x["schedule"]):
datas = x["schedule"][0]
datas = x["schedule"][0]
if len(datas) > 0:
approved_from = 'Запроваджено ' + datas['scheduleApprovedSince']
@ -259,10 +324,10 @@ def process_GPV(oe_account_number, tg_chat, tg_token):
hours_list = []
approved_from = 'Hемає даних про дату запровадження'
event_date = dt.today().strftime('%Y-%m-%d')
if IS_DEBUG:
print(f'hours_list{hours_list}, approved_from={approved_from}, event_date={event_date}')
for h in hours_list:
if h["status"] == 1:
hours_off.append(h["from"])
@ -282,6 +347,12 @@ def process_GPV(oe_account_number, tg_chat, tg_token):
if tg_response.status_code != 200:
sys.exit(STATUS_ERROR_SEND_TO_TG)
if USE_MATRIX:
res = send_matrix_notification(message)
if not res:
sys.exit(STATUS_ERROR_SEND_TO_MX)
def process_PO(apart, house, part_number, settlement, street, tg_chat, tg_token):
global IS_DEBUG
if not IS_DEBUG:
@ -294,11 +365,11 @@ def process_PO(apart, house, part_number, settlement, street, tg_chat, tg_token)
root = etree.parse(StringIO(html), parser)
else:
html = "<html><body><div id='shutdowns-table'><div class='table'><div class='flex table-header'><div class='region'>Район</div><div class='city'>Населений пункт</div><div class='street'>Вулиця</div><div class='house_number'>№ буд</div>" \
"<div class='building_part_number'>Корпус</div><div class='dispatch_name'>Диспетчерська назва об'єкту</div><div class='type-shutdown'>Вид робіт</div><div class='reason-shutdown'>Причина робіт</div><div class='placement_date'>Дата розміщення</div>" \
"<div class='shutdown_date'>Дата і час вимкнення</div><div class='turn_on_date'>Дата і час включення</div><div class='next_date'>Наступна дата робіт</div></div></div><div class='shutdowns'><div><div><div class='table-row flex'><div class='region'>Піднянський район</div>" \
"<div class='city'>Гірна</div><div class='street'>Тевського</div><div class='house_number'>9</div><div class='building_part_number'/><div class='dispatch_name'>ДСП 110/Пр.&quot;ЗБВIК&quot;/30-413/Л-6</div><div class='type-shutdown'>Планове</div>" \
"<div class='reason-shutdown reason-text-container'><div class='reason-text-hide'>Поточний ремонт</div></div><div class='placement_date'><div class='date'>2025.03.24</div><div class='time'>11:50</div></div><div class='shutdown_date'><div class='date'>2025.04.01</div>" \
"<div class='time'>09:00</div></div><div class='turn_on_date'><div class='date'>2025.04.01</div><div class='time'>18:00</div></div><div class='next_date'><div class='reason-text-hide'/></div></div></div></div></div></div></body></html>"
"<div class='building_part_number'>Корпус</div><div class='dispatch_name'>Диспетчерська назва об'єкту</div><div class='type-shutdown'>Вид робіт</div><div class='reason-shutdown'>Причина робіт</div><div class='placement_date'>Дата розміщення</div>" \
"<div class='shutdown_date'>Дата і час вимкнення</div><div class='turn_on_date'>Дата і час включення</div><div class='next_date'>Наступна дата робіт</div></div></div><div class='shutdowns'><div><div><div class='table-row flex'><div class='region'>Піднянський район</div>" \
"<div class='city'>Гірна</div><div class='street'>Тевського</div><div class='house_number'>9</div><div class='building_part_number'/><div class='dispatch_name'>ДСП 110/Пр.&quot;ЗБВIК&quot;/30-413/Л-6</div><div class='type-shutdown'>Планове</div>" \
"<div class='reason-shutdown reason-text-container'><div class='reason-text-hide'>Поточний ремонт</div></div><div class='placement_date'><div class='date'>2025.03.24</div><div class='time'>11:50</div></div><div class='shutdown_date'><div class='date'>2025.04.01</div>" \
"<div class='time'>09:00</div></div><div class='turn_on_date'><div class='date'>2025.04.01</div><div class='time'>18:00</div></div><div class='next_date'><div class='reason-text-hide'/></div></div></div></div></div></div></body></html>"
root = etree.fromstring(html)
message, need_to_send = get_message_with_PO(root=root, html=html)
@ -310,6 +381,12 @@ def process_PO(apart, house, part_number, settlement, street, tg_chat, tg_token)
if tg_response.status_code != 200:
sys.exit(STATUS_ERROR_SEND_TO_TG)
if USE_MATRIX:
res = send_matrix_notification(message)
if not res:
sys.exit(STATUS_ERROR_SEND_TO_MX)
def main():
global IS_DEBUG
tg_token, tg_chat, oe_account_number, IS_DEBUG, settlement, street, house, part_number, apart = read_config()
@ -323,5 +400,6 @@ def main():
process_PO(apart, house, part_number, settlement, street, tg_chat, tg_token)
if __name__ == "__main__":
main()

View File

@ -1,10 +1,14 @@
token: !secret token
chat_id: !secret chat_id
account: !secret account
debug: "true"
debug: "false"
settlement: !secret settlement
street: !secret street
house: "9"
building_part_number: ""
apartment: ""
mentions: !secret mentions
mentions: !secret mentions
use_matrix: "true"
mx_server: !secret mx_server
mx_room_id: !secret mx_room_id
mx_access_token: !secret mx_access_token

View File

@ -1,4 +1,4 @@
requests~=2.32.3
requests~=2.32.4
urllib3~=2.5.0
numpy~=2.2.0
lxml~=5.3.1