Несколько лет назад я собрал tk-скрипт, который прослушивал события с фрегата и показывал изображение с камеры на несколько секунд, когда кто-то был у двери. Это работало без особых усилий, пока я не обновился до F43 (python 3.14), теперь я застрял, глядя в окно, как один из тех людей, которые выходят на улицу!
Я подозреваю, что проблема находится где-то между запросами и многопоточностью, но я не знаю, как ее устранить. Кстати, я не пытаюсь выполнить запрос (здесь нет aiohttp или asyncio), просто поток для tk и поток для mqtt (чей обратный вызов находится там, где происходит запрос).
Это подозрение подогревается возможностью вызвать поток() из main(), и он работает, как и ожидалось, но вызов в любом месте из любого из обратных вызовов mqtt выдает это
File "/opt/cameras/./cams.py", line 33, in on_connect
stream("garage", 10)
~~~~~~^^^^^^^^
File "/opt/cameras/./cams.py", line 64, in stream
render = ImageTk.PhotoImage(load)
File "/home/htpc/.local/lib/python3.14/site-packages/PIL/ImageTk.py", line 129, in __init__
self.__photo = tkinter.PhotoImage(**kw)
~~~~~~~~~~~~~~~~~~^^^^^^
File "/usr/lib64/python3.14/tkinter/__init__.py", line 4301, in __init__
Image.__init__(self, 'photo', name, cnf, master, **kw)
~~~~~~~~~~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.14/tkinter/__init__.py", line 4248, in __init__
self.tk.call(('image', 'create', imgtype, name,) + options)
~~~~~~~~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
_tkinter.TclError: image type "photo" does not exist
cams.py
#!/usr/bin/env python3
"""
Listen for mqtt events from frigate and show them on the htpc
"""
import sys
import json
from threading import Thread
from io import BytesIO
from tkinter import Tk, Label
#from tkinter import *
from PIL import Image, ImageTk
import paho.mqtt.client as mqtt
import requests
FRIGATE_API = ""
MQTT_SERVER = ""
MQTT_USERNAME = ""
MQTT_PASSWORD = ""
size = 540,540
WIN = Tk()
WIN.overrideredirect(True)
LABEL = Label(WIN)
LABEL.place(x=0, y=0)
LABEL.pack()
def on_connect(client, userdata, flags, rc, properties):
"""
connect callback
"""
stream("", 10) #just here for testing
del userdata, flags, properties
print(f"MQTT connected {rc}")
client.subscribe("frigate/events")
def on_message(client, userdata, msg):
"""
message callback
"""
del client, userdata
event = msg.payload
event = json.loads(event.decode())
if event['before']['stationary'] is False:
if 'front' in event['before']['current_zones']:
stream(event['before']['camera'], 10)
def stream(camera, i):
"""
do the actual drawing
"""
url = f"{FRIGATE_API}/{camera}/latest.jpg"
if i == -1:
WIN.withdraw()
else:
WIN.deiconify()
frame = requests.get(url, timeout=1)
load = Image.open(BytesIO(frame.content))
load.thumbnail(size, Image.Resampling.LANCZOS)
render = ImageTk.PhotoImage(load)
LABEL.configure(image=render)
LABEL.image = render
WIN.after(1000, stream, camera, i-1)
def main():
"""
main
"""
client = mqtt.Client(mqtt.CallbackAPIVersion.VERSION2)
client.username_pw_set(MQTT_USERNAME, MQTT_PASSWORD)
client.on_connect = on_connect
client.on_message = on_message
client.connect(MQTT_SERVER, 1883, 60)
print("dispatch client thread")
client_loop = Thread(target=client.loop_forever, daemon=True)
client_loop.start()
print("start tk main loop")
WIN.mainloop()
if __name__ == '__main__':
sys.exit(main())
Подробнее здесь: https://stackoverflow.com/questions/798 ... -fedora-43
Tkinter TclError после обновления до Fedora 43 ⇐ Python
Программы на Python
-
Anonymous
1766453713
Anonymous
Несколько лет назад я собрал tk-скрипт, который прослушивал события с фрегата и показывал изображение с камеры на несколько секунд, когда кто-то был у двери. Это работало без особых усилий, пока я не обновился до F43 (python 3.14), теперь я застрял, глядя в окно, как один из тех людей, которые выходят на улицу!
Я подозреваю, что проблема находится где-то между запросами и многопоточностью, но я не знаю, как ее устранить. Кстати, я не пытаюсь выполнить запрос (здесь нет aiohttp или asyncio), просто поток для tk и поток для mqtt (чей обратный вызов находится там, где происходит запрос).
Это подозрение подогревается возможностью вызвать поток() из main(), и он работает, как и ожидалось, но вызов в любом месте из любого из обратных вызовов mqtt выдает это
File "/opt/cameras/./cams.py", line 33, in on_connect
stream("garage", 10)
~~~~~~^^^^^^^^
File "/opt/cameras/./cams.py", line 64, in stream
render = ImageTk.PhotoImage(load)
File "/home/htpc/.local/lib/python3.14/site-packages/PIL/ImageTk.py", line 129, in __init__
self.__photo = tkinter.PhotoImage(**kw)
~~~~~~~~~~~~~~~~~~^^^^^^
File "/usr/lib64/python3.14/tkinter/__init__.py", line 4301, in __init__
Image.__init__(self, 'photo', name, cnf, master, **kw)
~~~~~~~~~~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/lib64/python3.14/tkinter/__init__.py", line 4248, in __init__
self.tk.call(('image', 'create', imgtype, name,) + options)
~~~~~~~~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
_tkinter.TclError: image type "photo" does not exist
cams.py
#!/usr/bin/env python3
"""
Listen for mqtt events from frigate and show them on the htpc
"""
import sys
import json
from threading import Thread
from io import BytesIO
from tkinter import Tk, Label
#from tkinter import *
from PIL import Image, ImageTk
import paho.mqtt.client as mqtt
import requests
FRIGATE_API = ""
MQTT_SERVER = ""
MQTT_USERNAME = ""
MQTT_PASSWORD = ""
size = 540,540
WIN = Tk()
WIN.overrideredirect(True)
LABEL = Label(WIN)
LABEL.place(x=0, y=0)
LABEL.pack()
def on_connect(client, userdata, flags, rc, properties):
"""
connect callback
"""
stream("", 10) #just here for testing
del userdata, flags, properties
print(f"MQTT connected {rc}")
client.subscribe("frigate/events")
def on_message(client, userdata, msg):
"""
message callback
"""
del client, userdata
event = msg.payload
event = json.loads(event.decode())
if event['before']['stationary'] is False:
if 'front' in event['before']['current_zones']:
stream(event['before']['camera'], 10)
def stream(camera, i):
"""
do the actual drawing
"""
url = f"{FRIGATE_API}/{camera}/latest.jpg"
if i == -1:
WIN.withdraw()
else:
WIN.deiconify()
frame = requests.get(url, timeout=1)
load = Image.open(BytesIO(frame.content))
load.thumbnail(size, Image.Resampling.LANCZOS)
render = ImageTk.PhotoImage(load)
LABEL.configure(image=render)
LABEL.image = render
WIN.after(1000, stream, camera, i-1)
def main():
"""
main
"""
client = mqtt.Client(mqtt.CallbackAPIVersion.VERSION2)
client.username_pw_set(MQTT_USERNAME, MQTT_PASSWORD)
client.on_connect = on_connect
client.on_message = on_message
client.connect(MQTT_SERVER, 1883, 60)
print("dispatch client thread")
client_loop = Thread(target=client.loop_forever, daemon=True)
client_loop.start()
print("start tk main loop")
WIN.mainloop()
if __name__ == '__main__':
sys.exit(main())
Подробнее здесь: [url]https://stackoverflow.com/questions/79853316/tkinter-tclerror-after-update-to-fedora-43[/url]
Ответить
1 сообщение
• Страница 1 из 1
Перейти
- Кемерово-IT
- ↳ Javascript
- ↳ C#
- ↳ JAVA
- ↳ Elasticsearch aggregation
- ↳ Python
- ↳ Php
- ↳ Android
- ↳ Html
- ↳ Jquery
- ↳ C++
- ↳ IOS
- ↳ CSS
- ↳ Excel
- ↳ Linux
- ↳ Apache
- ↳ MySql
- Детский мир
- Для души
- ↳ Музыкальные инструменты даром
- ↳ Печатная продукция даром
- Внешняя красота и здоровье
- ↳ Одежда и обувь для взрослых даром
- ↳ Товары для здоровья
- ↳ Физкультура и спорт
- Техника - даром!
- ↳ Автомобилистам
- ↳ Компьютерная техника
- ↳ Плиты: газовые и электрические
- ↳ Холодильники
- ↳ Стиральные машины
- ↳ Телевизоры
- ↳ Телефоны, смартфоны, плашеты
- ↳ Швейные машинки
- ↳ Прочая электроника и техника
- ↳ Фототехника
- Ремонт и интерьер
- ↳ Стройматериалы, инструмент
- ↳ Мебель и предметы интерьера даром
- ↳ Cантехника
- Другие темы
- ↳ Разное даром
- ↳ Давай меняться!
- ↳ Отдам\возьму за копеечку
- ↳ Работа и подработка в Кемерове
- ↳ Давай с тобой поговорим...
Мобильная версия