{"id":476858,"date":"2026-04-21T16:27:21","date_gmt":"2026-04-21T16:27:21","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=476858"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=476858","title":{"rendered":"\u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u0437\u0430\u0433\u0440\u0443\u0437\u0447\u0438\u043a \u0432\u0438\u0434\u0435\u043e \u043d\u0430 aiogram 3 \u0438 yt-dlp: \u043a\u0430\u043a \u043d\u0435 \u043f\u043e\u043b\u043e\u0436\u0438\u0442\u044c Event Loop \u0438 \u043f\u0440\u0438\u043a\u0440\u0443\u0442\u0438\u0442\u044c \u0447\u0435\u0441\u0442\u043d\u044b\u0439 \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0431\u0430\u0440"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/f89\/3f4\/4da\/f893f44da60b27b283c536d9f18af408.jpg\" alt=\"\u0431\u043e\u0442 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431, \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c, \u0442\u0438\u043a\u0442\u043e\u043a\" title=\"\u0431\u043e\u0442 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431, \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c, \u0442\u0438\u043a\u0442\u043e\u043a\" width=\"1280\" height=\"722\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/f89\/3f4\/4da\/f893f44da60b27b283c536d9f18af408.jpg 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/f89\/3f4\/4da\/f893f44da60b27b283c536d9f18af408.jpg 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption>\u0431\u043e\u0442 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431, \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c, \u0442\u0438\u043a\u0442\u043e\u043a<\/figcaption><\/div>\n<\/figure>\n<h3>\u0417\u0430\u0447\u0435\u043c \u0432 2026 \u0433\u043e\u0434\u0443 \u043f\u0438\u0441\u0430\u0442\u044c \u0435\u0449\u0451 \u043e\u0434\u0438\u043d \u0437\u0430\u0433\u0440\u0443\u0437\u0447\u0438\u043a \u0432\u0438\u0434\u0435\u043e<\/h3>\n<p>\u041a\u0430\u0437\u0430\u043b\u043e\u0441\u044c \u0431\u044b, \u0442\u0435\u043c\u0430 \u0437\u0430\u0435\u0437\u0436\u0435\u043d\u043d\u0430\u044f \u0434\u043e \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u043f\u043e\u043a\u0440\u044b\u0448\u043a\u0438 \u043a\u0430\u043c\u0430\u0437\u0430: \u00ab\u0441\u0434\u0435\u043b\u0430\u0439 \u0431\u043e\u0442\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043a\u0430\u0447\u0430\u0435\u0442 \u0440\u043e\u043b\u0438\u043a\u0438\u00bb. \u041d\u0430 \u0425\u0430\u0431\u0440\u0435 \u0443\u0436\u0435 \u043b\u0435\u0436\u0438\u0442 \u0434\u0435\u0441\u044f\u0442\u043e\u043a \u0442\u0443\u0442\u043e\u0440\u0438\u0430\u043b\u043e\u0432, \u043d\u0430 GitHub \u2014 \u0441\u043e\u0442\u043d\u0438 \u0444\u043e\u0440\u043a\u043e\u0432. \u041d\u043e \u0435\u0441\u043b\u0438 \u0432\u044b \u0445\u043e\u0442\u044c \u0440\u0430\u0437 \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u043b\u0438 \u0442\u0430\u043a\u043e\u0439 \u0431\u043e\u0442 \u0432 \u043f\u0440\u043e\u0434\u0435 \u2014 \u0432\u044b \u0437\u043d\u0430\u0435\u0442\u0435, \u0447\u0442\u043e 90% \u0438\u0437 \u043d\u0438\u0445 \u043f\u0430\u0434\u0430\u044e\u0442 \u043d\u0430 \u0432\u0442\u043e\u0440\u043e\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435, \u0430 \u043e\u0441\u0442\u0430\u0432\u0448\u0438\u0435\u0441\u044f 10% \u0447\u0435\u0441\u0442\u043d\u043e \u0443\u043c\u0438\u0440\u0430\u044e\u0442 \u043d\u0430 \u0432\u0438\u0434\u0435\u043e \u0434\u043b\u0438\u043d\u043d\u0435\u0435 15 \u043c\u0438\u043d\u0443\u0442.<\/p>\n<p>\u042f \u043f\u043e\u043b\u0435\u0437 \u0432 \u043e\u0440\u0433\u0430\u043d\u0438\u0447\u0435\u0441\u043a\u0443\u044e \u0432\u044b\u0434\u0430\u0447\u0443 Google, \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u043d\u044f\u0442\u044c, <strong>\u043f\u043e\u0447\u0435\u043c\u0443<\/strong> \u043b\u044e\u0434\u0438 \u0432\u043e\u043e\u0431\u0449\u0435 \u0438\u0449\u0443\u0442 Telegram-\u0431\u043e\u0442\u043e\u0432, \u0430 \u043d\u0435 \u043f\u043e\u043b\u044c\u0437\u0443\u044e\u0442\u0441\u044f \u0432\u0435\u0431-\u0441\u0435\u0440\u0432\u0438\u0441\u0430\u043c\u0438. \u0418 \u0432\u044b\u0434\u0430\u0447\u0430, \u0447\u0435\u0441\u0442\u043d\u043e \u0433\u043e\u0432\u043e\u0440\u044f, \u0443\u0434\u0440\u0443\u0447\u0430\u0435\u0442.<\/p>\n<h4>\u041c\u0438\u043d\u0438-\u0430\u043d\u0430\u043b\u0438\u0442\u0438\u043a\u0430 \u043f\u043e\u0438\u0441\u043a\u043e\u0432\u043e\u0439 \u0432\u044b\u0434\u0430\u0447\u0438 (\u0438\u043b\u0438 \u043f\u043e\u0447\u0435\u043c\u0443 \u043b\u044e\u0434\u0438 \u0438\u0434\u0443\u0442 \u0432 Telegram)<\/h4>\n<p>\u041a\u043e\u0433\u0434\u0430 \u044f \u0438\u0437 \u043b\u044e\u0431\u043e\u043f\u044b\u0442\u0441\u0442\u0432\u0430 \u043f\u043e\u0448\u0451\u043b \u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c, \u0447\u0442\u043e \u043d\u0430 \u0441\u0430\u043c\u043e\u043c \u0434\u0435\u043b\u0435 \u0433\u0443\u0433\u043b\u044f\u0442 \u0436\u0438\u0432\u044b\u0435 \u043b\u044e\u0434\u0438 \u0432\u043e\u043a\u0440\u0443\u0433 \u0442\u0435\u043c\u044b \u00ab\u0441\u043a\u0430\u0447\u0430\u0442\u044c \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431\u0430\u00bb, \u043a\u0430\u0440\u0442\u0438\u043d\u0430 \u0441\u043b\u043e\u0436\u0438\u043b\u0430\u0441\u044c \u0434\u043e\u0432\u043e\u043b\u044c\u043d\u043e \u043f\u0440\u0435\u0434\u0441\u043a\u0430\u0437\u0443\u0435\u043c\u0430\u044f. \u0411\u0430\u0437\u043e\u0432\u044b\u0439 \u0437\u0430\u043f\u0440\u043e\u0441 \u2014 \u044d\u0442\u043e \u0431\u0430\u043d\u0430\u043b\u044c\u043d\u043e\u0435 \u00ab\u0441\u043a\u0430\u0447\u0430\u0442\u044c \u0432\u0438\u0434\u0435\u043e \u0441 YouTube \u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0435\u00bb, \u0441 \u0432\u0430\u0440\u0438\u0430\u0446\u0438\u044f\u043c\u0438 \u0440\u0430\u0441\u043a\u043b\u0430\u0434\u043a\u0438 \u0438 \u043e\u043f\u0435\u0447\u0430\u0442\u043e\u043a \u0443\u0440\u043e\u0432\u043d\u044f \u00ab\u044e\u0442\u0443\u0431 com\u00bb. \u0414\u0430\u043b\u044c\u0448\u0435 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u0441\u0435\u0433\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044f \u043f\u043e \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0443: \u043a\u043e\u043c\u0443-\u0442\u043e \u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u043d\u0443\u0436\u043d\u043e 1080p (\u043f\u0440\u0438\u0447\u0451\u043c \u043f\u043e\u043b\u043e\u0432\u0438\u043d\u0430 \u043f\u0438\u0448\u0435\u0442 \u00ab1080\u0440\u00bb \u043a\u0438\u0440\u0438\u043b\u043b\u0438\u0446\u0435\u0439, \u0434\u0430\u0436\u0435 \u043d\u0435 \u0437\u0430\u043c\u0435\u0447\u0430\u044f \u044d\u0442\u043e\u0433\u043e), \u0430 \u043a\u043e\u043c\u0443-\u0442\u043e \u2014 \u0433\u043e\u043b\u044b\u0439 MP4-\u0444\u0430\u0439\u043b \u0431\u0435\u0437 \u043b\u0438\u0448\u043d\u0438\u0445 \u043e\u0431\u0432\u0435\u0441\u043e\u0432.<\/p>\n<p>\u0412\u0442\u043e\u0440\u0430\u044f \u043e\u0441\u044c \u2014 \u0441\u043f\u043e\u0441\u043e\u0431 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f. \u0427\u0435\u043b\u043e\u0432\u0435\u043a \u0445\u043e\u0447\u0435\u0442 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e, \u043e\u043d\u043b\u0430\u0439\u043d \u0438 \u0436\u0435\u043b\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u0435\u0437 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438 (\u0432 2026-\u043c \u0444\u043e\u0440\u043c\u0430 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438 \u043f\u0443\u0433\u0430\u0435\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0441\u0438\u043b\u044c\u043d\u0435\u0435, \u0447\u0435\u043c \u0431\u0430\u043d\u043d\u0435\u0440 \u043e\u043d\u043b\u0430\u0439\u043d-\u043a\u0430\u0437\u0438\u043d\u043e, \u0438 \u044f \u0435\u0433\u043e \u043f\u043e\u043d\u0438\u043c\u0430\u044e). \u0418\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u0434 \u044d\u0442\u043e\u0442 \u0438\u043d\u0442\u0435\u043d\u0442 \u0437\u0430\u0442\u043e\u0447\u0435\u043d\u044b \u043f\u0435\u0440\u0432\u044b\u0435 \u0434\u0435\u0441\u044f\u0442\u044c \u0441\u0442\u0440\u043e\u043a \u0432\u044b\u0434\u0430\u0447\u0438: \u0441\u043d\u0430\u0440\u0443\u0436\u0438 \u2014 \u0430\u043a\u043a\u0443\u0440\u0430\u0442\u043d\u044b\u0439 \u043b\u0435\u043d\u0434\u0438\u043d\u0433 \u0441 \u0431\u043e\u043b\u044c\u0448\u043e\u0439 \u043a\u043d\u043e\u043f\u043a\u043e\u0439, \u0432\u043d\u0443\u0442\u0440\u0438 \u2014 \u0442\u0440\u0438 \u0440\u0435\u0434\u0438\u0440\u0435\u043a\u0442\u0430, \u043f\u043e\u043f\u0430\u043f-\u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0449\u0438\u043a, \u0438\u043d\u0441\u0442\u0430\u043b\u043b\u0435\u0440 <code>Yandex.Browser Setup.exe<\/code> \u0438, \u0435\u0441\u043b\u0438 \u043f\u043e\u0432\u0435\u0437\u0451\u0442, \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 \u043c\u0430\u0439\u043d\u0435\u0440 \u0432 \u043f\u043e\u0434\u0430\u0440\u043e\u043a. \u0421\u0443\u0442\u044c \u0432\u0441\u0435\u0445 \u044d\u0442\u0438\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432, \u043a\u0430\u043a \u043d\u0438 \u043a\u0440\u0443\u0442\u0438 \u0438\u0445 \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438, \u0441\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u043a \u043e\u0434\u043d\u043e\u043c\u0443 \u2014 \u043d\u0430\u0436\u0430\u0442\u044c \u043e\u0434\u043d\u0443 \u043a\u043d\u043e\u043f\u043a\u0443 \u0438 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c MP4 \u0432 \u0445\u043e\u0440\u043e\u0448\u0435\u043c \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0435. \u0412\u0441\u0451 \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u043e\u0435 \u2014 \u0443\u0436\u0435 \u043d\u0430\u0448\u0438 \u0441 \u0432\u0430\u043c\u0438 \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u043d\u044b\u0435 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b. \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u043e\u0430\u0442\u0435\u043b\u044c \u043d\u0435 \u0445\u043e\u0447\u0435\u0442:<\/p>\n<ul>\n<li>\n<p>\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u0440\u0435\u043a\u043b\u0430\u043c\u0443 \u043a\u0430\u0437\u0438\u043d\u043e \u00ab\u041f\u0438\u043d-\u0410\u043f\u00bb;<\/p>\n<\/li>\n<li>\n<p>\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u00ab\u043f\u043b\u0430\u0433\u0438\u043d \u0434\u043b\u044f \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u00bb;<\/p>\n<\/li>\n<li>\n<p>\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c\u0441\u044f, \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0430\u0442\u044c \u043f\u043e\u0447\u0442\u0443 \u0438 \u043f\u0440\u043e\u0445\u043e\u0434\u0438\u0442\u044c \u043a\u0430\u043f\u0447\u0443 \u0441 \u043f\u043e\u0436\u0430\u0440\u043d\u044b\u043c\u0438 \u0433\u0438\u0434\u0440\u0430\u043d\u0442\u0430\u043c\u0438;<\/p>\n<\/li>\n<li>\n<p>\u043e\u0431\u044a\u044f\u0441\u043d\u044f\u0442\u044c \u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441\u0443, \u0447\u0442\u043e <code>setup_downloader_pro_final2.exe<\/code> \u2014 \u044d\u0442\u043e \u044f\u043a\u043e\u0431\u044b \u043b\u0435\u0433\u0438\u0442\u0438\u043c\u043d\u043e.<\/p>\n<\/li>\n<\/ul>\n<p>\u0418\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u044d\u0442\u043e\u043c\u0443 <strong>Telegram-\u0431\u043e\u0442<\/strong> \u0432\u044b\u0438\u0433\u0440\u044b\u0432\u0430\u0435\u0442 \u0443 \u0432\u0435\u0431-\u0441\u0435\u0440\u0432\u0438\u0441\u043e\u0432 \u043a\u0430\u043a \u043a\u043e\u043d\u0446\u0435\u043f\u0442: \u043d\u0435\u0442 \u0440\u0435\u043a\u043b\u0430\u043c\u044b, \u043d\u0435\u0442 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438, \u043d\u0435\u0442 \u0441\u0442\u0440\u0430\u043d\u043d\u044b\u0445 exe-\u0448\u043d\u0438\u043a\u043e\u0432. \u041f\u0440\u0438\u0441\u043b\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u043b \u0444\u0430\u0439\u043b. \u0412\u0441\u0451.<\/p>\n<p>\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u043e\u0434\u043d\u0430: <strong>\u043d\u0430\u043f\u0438\u0441\u0430\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0431\u043e\u0442 \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u043e \u2014 \u0441\u043b\u043e\u0436\u043d\u0435\u0435, \u0447\u0435\u043c \u043a\u0430\u0436\u0435\u0442\u0441\u044f<\/strong>. \u041e\u0431 \u044d\u0442\u043e\u043c \u0438 \u0441\u0442\u0430\u0442\u044c\u044f.<\/p>\n<hr\/>\n<h3>\u0421\u0442\u0435\u043a: \u043f\u043e\u0447\u0435\u043c\u0443 Python + aiogram 3 + yt-dlp<\/h3>\n<p>\u041a\u043e\u0440\u043e\u0442\u043a\u043e \u0438 \u043f\u043e \u0434\u0435\u043b\u0443, \u0431\u0435\u0437 \u043e\u0447\u0435\u0440\u0435\u0434\u043d\u043e\u0433\u043e \u0441\u0440\u0430\u0432\u043d\u0435\u043d\u0438\u044f \u0441 Go \u0438 Node.<\/p>\n<ul>\n<li>\n<p><strong>Python 3.11+<\/strong> \u2014 \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e <code>asyncio<\/code> \u0432 11-\u0439 \u0432\u0435\u0442\u043a\u0435 \u0441\u0442\u0430\u043b \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u044b\u0441\u0442\u0440\u044b\u043c, <code>TaskGroup<\/code> \u043f\u0440\u0438\u0435\u0445\u0430\u043b, \u0430 <code>tomllib<\/code> \u0432\u0441\u0442\u0440\u043e\u0435\u043d.<\/p>\n<\/li>\n<li>\n<p><strong>aiogram 3.x<\/strong> \u2014 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0439 \u0440\u043e\u0443\u0442\u0435\u0440, <code>Dispatcher<\/code> \u043d\u0430 DI, FSM \u0438\u0437 \u043a\u043e\u0440\u043e\u0431\u043a\u0438, \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u044b\u0439 Type Hinting, middlewares, \u0444\u0438\u043b\u044c\u0442\u0440\u044b. \u042d\u0442\u043e \u043d\u0435 <code>python-telegram-bot<\/code> \u0432 \u0435\u0433\u043e \u043a\u043b\u0430\u0441\u0441\u0438\u0447\u0435\u0441\u043a\u043e\u043c \u043f\u0435\u0440\u0435\u0433\u0440\u0443\u0436\u0435\u043d\u043d\u043e\u043c \u0432\u0438\u0434\u0435 \u0438 \u043d\u0435 telebot \u043e\u0431\u0440\u0430\u0437\u0446\u0430 2018-\u0433\u043e.<\/p>\n<\/li>\n<li>\n<p><strong>yt-dlp<\/strong> \u2014 \u0444\u043e\u0440\u043a <code>youtube-dl<\/code>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0436\u0438\u0432\u0435\u0435 \u0432\u0441\u0435\u0445 \u0436\u0438\u0432\u044b\u0445. \u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e YouTube, \u043d\u043e \u0438 \u043e\u043a\u043e\u043b\u043e <strong>1500 \u0441\u0430\u0439\u0442\u043e\u0432<\/strong> (TikTok, VK, Instagram, X\/Twitter, Vimeo, Rutube \u0438 \u0442\u0430\u043a \u0434\u0430\u043b\u0435\u0435). \u041f\u0430\u0440\u0441\u0435\u0440\u044b \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u044e\u0442\u0441\u044f \u0447\u0430\u0449\u0435, \u0447\u0435\u043c \u044f \u043a\u043e\u043c\u043c\u0438\u0447\u0443.<\/p>\n<\/li>\n<\/ul>\n<p>\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043d\u043e \u0437\u0430\u0434\u0430\u0447\u0430 \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u043e: \u043f\u0440\u0438\u043d\u044f\u043b \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u2192 \u0432\u044b\u0434\u0440\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u2192 \u0432\u044b\u0437\u0432\u0430\u043b <code>yt-dlp<\/code> \u2192 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u043b \u0444\u0430\u0439\u043b. \u0418 \u0432\u043e\u0442 \u0437\u0434\u0435\u0441\u044c \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u0441\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435.<\/p>\n<hr\/>\n<h3>\u0413\u043b\u0430\u0432\u043d\u0430\u044f \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430: \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0439 IO \u0438 \u0441\u043c\u0435\u0440\u0442\u044c Event Loop<\/h3>\n<p>\u041d\u043e\u0432\u0438\u0447\u043e\u043a, \u043d\u0430\u043f\u0438\u0441\u0430\u0432\u0448\u0438\u0439 \u043f\u0435\u0440\u0432\u043e\u0433\u043e \u0431\u043e\u0442\u0430 \u043d\u0430 <code>aiogram<\/code>, \u0434\u0435\u043b\u0430\u0435\u0442 \u043f\u0440\u0438\u043c\u0435\u0440\u043d\u043e \u0442\u0430\u043a:<\/p>\n<p>python         <\/p>\n<pre><code class=\"python\">@router.message(F.text.startswith(\"http\"))async def download_handler(message: Message) -&gt; None:    url = message.text    with YoutubeDL({\"outtmpl\": \"video.mp4\"}) as ydl:        ydl.download([url])  # \ud83d\udd25 \u0442\u0443\u0442 \u0432\u0441\u0451 \u0438 \u0443\u043c\u0438\u0440\u0430\u0435\u0442    await message.answer_video(FSInputFile(\"video.mp4\"))<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:87px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041d\u0430 \u043e\u0434\u043d\u043e\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435 \u044d\u0442\u043e \u0434\u0430\u0436\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442. \u041d\u0430 \u0432\u0442\u043e\u0440\u043e\u043c \u2014 \u0431\u043e\u0442 \u043f\u0440\u0435\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0432 \u0442\u044b\u043a\u0432\u0443.<\/p>\n<h4>\u041f\u043e\u0447\u0435\u043c\u0443 \u0442\u0430\u043a \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442<\/h4>\n<p><code>asyncio<\/code> \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0432 <strong>\u043e\u0434\u043d\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435<\/strong> \u2014 \u044d\u0442\u043e \u043a\u043e\u043e\u043f\u0435\u0440\u0430\u0442\u0438\u0432\u043d\u0430\u044f \u043c\u043d\u043e\u0433\u043e\u0437\u0430\u0434\u0430\u0447\u043d\u043e\u0441\u0442\u044c. Event Loop \u0436\u043e\u043d\u0433\u043b\u0438\u0440\u0443\u0435\u0442 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0430\u043c\u0438, \u043f\u0435\u0440\u0435\u043a\u043b\u044e\u0447\u0430\u044f\u0441\u044c \u043c\u0435\u0436\u0434\u0443 \u043d\u0438\u043c\u0438 \u043d\u0430 <code>await<\/code>. \u041f\u043e\u043a\u0430 \u043e\u0434\u043d\u0430 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0430 \u043d\u0435 \u0432\u0441\u0442\u0440\u0435\u0442\u0438\u043b\u0430 <code>await<\/code> \u0438\u043b\u0438 \u043d\u0435 \u0437\u0430\u0432\u0435\u0440\u0448\u0438\u043b\u0430\u0441\u044c \u2014 \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u044b\u0435 \u0436\u0434\u0443\u0442.<\/p>\n<p><code>yt-dlp<\/code> \u2014 <strong>\u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u0430\u044f<\/strong> \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430. \u041f\u043e\u0434 \u043a\u0430\u043f\u043e\u0442\u043e\u043c \u043e\u043d\u0430:<\/p>\n<ol>\n<li>\n<p>\u0425\u043e\u0434\u0438\u0442 HTTP-\u0437\u0430\u043f\u0440\u043e\u0441\u0430\u043c\u0438 \u0447\u0435\u0440\u0435\u0437 <code>urllib<\/code> (\u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0439 \u0441\u043e\u043a\u0435\u0442).<\/p>\n<\/li>\n<li>\n<p>\u041f\u0430\u0440\u0441\u0438\u0442 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b, \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u0442 URL \u0441\u0442\u0440\u0438\u043c\u043e\u0432.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043a\u0430\u0447\u0438\u0432\u0430\u0435\u0442 \u0447\u0430\u043d\u043a\u0438 \u0447\u0435\u0440\u0435\u0437 \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0438 IO.<\/p>\n<\/li>\n<li>\n<p>\u0412\u044b\u0437\u044b\u0432\u0430\u0435\u0442 <code>ffmpeg<\/code> \u0447\u0435\u0440\u0435\u0437 <code>subprocess<\/code> \u2014 \u044d\u0442\u043e \u0442\u043e\u0436\u0435 \u0431\u043b\u043e\u043a.<\/p>\n<\/li>\n<\/ol>\n<p>\u041a\u043e\u0433\u0434\u0430 \u0432\u044b \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0435 <a href=\"http:\/\/ydl.download\" rel=\"noopener noreferrer nofollow\"><code>ydl.download<\/code><\/a><code>([url])<\/code> \u0432 <code>async def<\/code>-\u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0435, \u0432\u044b \u0431\u0443\u043a\u0432\u0430\u043b\u044c\u043d\u043e <strong>\u0437\u0430\u043c\u043e\u0440\u0430\u0436\u0438\u0432\u0430\u0435\u0442\u0435 Event Loop<\/strong> \u043d\u0430 \u0432\u0441\u0451 \u0432\u0440\u0435\u043c\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f. \u041f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u044c\u0442\u0435: \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0410 \u043f\u0440\u0438\u0441\u043b\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 2-\u0447\u0430\u0441\u043e\u0432\u043e\u0439 \u0441\u0442\u0440\u0438\u043c \u0432 4K. \u041d\u0430 2 \u0447\u0430\u0441\u0430 \u0432\u0430\u0448 \u0431\u043e\u0442 <strong>\u043f\u0435\u0440\u0435\u0441\u0442\u0430\u0451\u0442 \u043e\u0442\u0432\u0435\u0447\u0430\u0442\u044c<\/strong> \u0430\u0431\u0441\u043e\u043b\u044e\u0442\u043d\u043e \u0432\u0441\u0435\u043c \u2014 \u043a\u043d\u043e\u043f\u043a\u0438 \u043d\u0435 \u043d\u0430\u0436\u0438\u043c\u0430\u044e\u0442\u0441\u044f, <code>\/start<\/code> \u0438\u0433\u043d\u043e\u0440\u0438\u0440\u0443\u0435\u0442\u0441\u044f, middlewares \u043c\u043e\u043b\u0447\u0430\u0442. \u0418 \u043d\u0435\u0442, GIL \u0437\u0434\u0435\u0441\u044c \u043d\u0435 \u043f\u043e\u043c\u043e\u0436\u0435\u0442: \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u0434\u0430\u0436\u0435 \u043d\u0435 \u0432 \u043d\u0451\u043c, \u0430 \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u0443 \u0432\u0430\u0441 <strong>\u043e\u0434\u0438\u043d \u043f\u043e\u0442\u043e\u043a \u0434\u043b\u044f \u0432\u0441\u0435\u0439 \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e\u0439 \u043c\u0430\u0448\u0438\u043d\u044b<\/strong>.<\/p>\n<p>\u041c\u043d\u043e\u0433\u0438\u0435 \u043d\u0430\u0438\u0432\u043d\u043e \u0434\u0443\u043c\u0430\u044e\u0442: \u00ab\u043d\u0443 \u0442\u0430\u043a GIL \u0436\u0435 \u0432\u0441\u0451 \u0440\u0430\u0432\u043d\u043e \u043e\u0442\u043f\u0443\u0441\u043a\u0430\u0435\u0442\u0441\u044f \u043d\u0430 IO, \u043f\u0440\u043e\u0431\u043b\u0435\u043c \u0431\u044b\u0442\u044c \u043d\u0435 \u0434\u043e\u043b\u0436\u043d\u043e\u00bb. \u041f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u0435\u0441\u0442\u044c \u2014 GIL \u043e\u0442\u043f\u0443\u0441\u043a\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430 CPython, \u043d\u043e <code>asyncio<\/code> \u043f\u0440\u043e \u044d\u0442\u043e \u043d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u0437\u043d\u0430\u0435\u0442. \u0414\u043b\u044f Event Loop \u043b\u044e\u0431\u043e\u0439 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u0432\u044b\u0437\u043e\u0432 \u2014 \u044d\u0442\u043e \u0447\u0451\u0440\u043d\u044b\u0439 \u044f\u0449\u0438\u043a, \u0438\u0437 \u043a\u043e\u0442\u043e\u0440\u043e\u0433\u043e \u043d\u0435 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435.<\/p>\n<p><strong>\u0412\u044b\u0432\u043e\u0434<\/strong>: \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u043a\u043e\u0434 \u0432 async-\u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0435 = DoS \u0441\u0430\u043c\u043e\u0433\u043e \u0441\u0435\u0431\u044f.<\/p>\n<hr\/>\n<h3>\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043d\u043e\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435: \u0432\u044b\u043d\u043e\u0441\u0438\u043c \u0440\u0430\u0431\u043e\u0442\u0443 \u0432 \u043f\u043e\u0442\u043e\u043a<\/h3>\n<p>\u0420\u0435\u0448\u0435\u043d\u0438\u0439 \u0443 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u0442\u0440\u0438:<\/p>\n<ol>\n<li>\n<p><a href=\"http:\/\/asyncio.to\" rel=\"noopener noreferrer nofollow\"><code><strong>asyncio.to<\/strong><\/code><\/a><code><strong>_thread()<\/strong><\/code> \u2014 \u0441\u0430\u0445\u0430\u0440 \u043f\u043e\u0432\u0435\u0440\u0445 <code>run_in_executor<\/code> \u0441 \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u043c \u043f\u0443\u043b\u043e\u043c. Pythonic, \u043b\u0435\u043d\u0438\u0432, \u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442 \u0434\u043b\u044f 95% \u0441\u043b\u0443\u0447\u0430\u0435\u0432.<\/p>\n<\/li>\n<li>\n<p><strong>\u041a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 <\/strong><code><strong>ThreadPoolExecutor<\/strong><\/code> \u2014 \u043a\u043e\u0433\u0434\u0430 \u043d\u0443\u0436\u0435\u043d \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e\u043c \u0432\u043e\u0440\u043a\u0435\u0440\u043e\u0432 \u0438 \u043e\u0447\u0435\u0440\u0435\u0434\u044c\u044e.<\/p>\n<\/li>\n<li>\n<p><code><strong>ProcessPoolExecutor<\/strong><\/code> \u2014 \u043a\u043e\u0433\u0434\u0430 \u0435\u0441\u0442\u044c \u0440\u0435\u0430\u043b\u044c\u043d\u044b\u0439 CPU-bound (\u043d\u0430\u043c \u043d\u0435 \u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442, <code>yt-dlp<\/code> \u2014 \u044d\u0442\u043e IO-bound).<\/p>\n<\/li>\n<\/ol>\n<p>\u042f \u0431\u0435\u0440\u0443 \u0432\u0430\u0440\u0438\u0430\u043d\u0442 \u21162 \u2014 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043f\u0443\u043b. \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u0435\u0441\u043b\u0438 \u0432\u044b \u043f\u0443\u0441\u0442\u0438\u0442\u0435 \u0432\u0441\u0451 \u0432 \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u0439 \u043f\u0443\u043b \u0447\u0435\u0440\u0435\u0437 <code>to_thread<\/code>, \u0442\u043e \u043f\u0440\u0438 50 \u043f\u0430\u0440\u0430\u043b\u043b\u0435\u043b\u044c\u043d\u044b\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u0430\u0445 \u0443 \u0432\u0430\u0441 \u0443\u043b\u0435\u0442\u0438\u0442 \u0432 \u043d\u0435\u0431\u0435\u0441\u0430 \u0438 \u043f\u0430\u043c\u044f\u0442\u044c, \u0438 \u0441\u0435\u0442\u044c. \u041d\u0443\u0436\u0435\u043d <strong>rate limiting \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b<\/strong>.<\/p>\n<h4>\u0421\u0435\u0440\u0432\u0438\u0441\u043d\u044b\u0439 \u0441\u043b\u043e\u0439 DownloaderService<\/h4>\n<p>python         <\/p>\n<pre><code class=\"python\">from __future__ import annotationsimport asyncioimport loggingfrom concurrent.futures import ThreadPoolExecutorfrom dataclasses import dataclassfrom pathlib import Pathfrom typing import Any, Callablefrom yt_dlp import YoutubeDLfrom yt_dlp.utils import DownloadErrorlogger = logging.getLogger(__name__)@dataclass(slots=True, frozen=True)class DownloadResult:    file_path: Path    title: str    duration: int    filesize: intclass DownloaderService:    \"\"\"    \u0421\u0435\u0440\u0432\u0438\u0441 \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e. \u0412\u0441\u044e \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0443\u044e \u0440\u0430\u0431\u043e\u0442\u0443 \u0432\u044b\u043d\u043e\u0441\u0438\u0442    \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0439 ThreadPoolExecutor, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043c\u0435\u0448\u0430\u0442\u044c Event Loop.    \"\"\"    def __init__(        self,        download_dir: Path,        max_workers: int = 4,    ) -&gt; None:        self._download_dir = download_dir        self._download_dir.mkdir(parents=True, exist_ok=True)        # \u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0438\u0432\u0430\u0435\u043c \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u0439.        # \u0411\u043e\u043b\u044c\u0448\u0435 4-8 \u0441\u043c\u044b\u0441\u043b\u0430 \u043d\u0435 \u0438\u043c\u0435\u0435\u0442: \u0443\u043f\u0440\u0451\u043c\u0441\u044f \u0432 \u0441\u0435\u0442\u044c\/\u0434\u0438\u0441\u043a\/RAM.        self._executor = ThreadPoolExecutor(            max_workers=max_workers,            thread_name_prefix=\"ytdlp-worker\",        )    async def download(        self,        url: str,        progress_hook: Callable[[dict[str, Any]], None] | None = None,    ) -&gt; DownloadResult:        \"\"\"        \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u0430\u044f \u043e\u0431\u0451\u0440\u0442\u043a\u0430. \u0412\u043d\u0443\u0442\u0440\u0438 \u2014 \u0447\u0435\u0441\u0442\u043d\u044b\u0439 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 yt-dlp,        \u043d\u043e \u0438\u0441\u043f\u043e\u043b\u043d\u044f\u0435\u0442\u0441\u044f \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435.        \"\"\"        loop = asyncio.get_running_loop()        return await loop.run_in_executor(            self._executor,            self._blocking_download,            url,            progress_hook,        )    def _blocking_download(        self,        url: str,        progress_hook: Callable[[dict[str, Any]], None] | None,    ) -&gt; DownloadResult:        \"\"\"\u042d\u0442\u043e\u0442 \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0435\u0442\u0441\u044f \u0412\u041d\u0415 event loop, \u0432 \u0440\u0430\u0431\u043e\u0447\u0435\u043c \u043f\u043e\u0442\u043e\u043a\u0435.\"\"\"        ydl_opts: dict[str, Any] = {            # \u041b\u0443\u0447\u0448\u0435\u0435 mp4 \u0434\u043e 1080p + \u043b\u0443\u0447\u0448\u0438\u0439 m4a, \u0441\u043c\u0451\u0440\u0436\u0435\u043d\u043d\u044b\u0435 \u0432 mp4.            \"format\": \"bv*[height&lt;=1080][ext=mp4]+ba[ext=m4a]\/b[ext=mp4]\/b\",            \"merge_output_format\": \"mp4\",            \"outtmpl\": str(self._download_dir \/ \"%(id)s.%(ext)s\"),            \"quiet\": True,            \"no_warnings\": True,            \"noprogress\": True,  # \u0441\u0432\u043e\u0439 \u0445\u0443\u043a, \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 stdout \u043d\u0435 \u043d\u0443\u0436\u0435\u043d            \"concurrent_fragment_downloads\": 4,            \"retries\": 3,        }        if progress_hook is not None:            ydl_opts[\"progress_hooks\"] = [progress_hook]        try:            with YoutubeDL(ydl_opts) as ydl:                info = ydl.extract_info(url, download=True)                file_path = Path(ydl.prepare_filename(info))        except DownloadError as e:            logger.warning(\"yt-dlp error for %s: %s\", url, e)            raise        return DownloadResult(            file_path=file_path,            title=info.get(\"title\", \"video\"),            duration=int(info.get(\"duration\") or 0),            filesize=file_path.stat().st_size,        )    async def shutdown(self) -&gt; None:        self._executor.shutdown(wait=True, cancel_futures=False)<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0427\u0442\u043e \u0437\u0434\u0435\u0441\u044c \u0432\u0430\u0436\u043d\u043e:<\/p>\n<ul>\n<li>\n<p><code><strong>run_in_executor<\/strong><\/code><strong> \u0441 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u043c \u043f\u0443\u043b\u043e\u043c<\/strong> \u2014 \u043c\u044b \u043d\u0435 \u0442\u0440\u043e\u0433\u0430\u0435\u043c \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u0439 executor, \u043e\u043d \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0434\u0440\u0443\u0433\u0438\u0445 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439 (<code>to_thread<\/code> \u0443 aiogram \u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a).<\/p>\n<\/li>\n<li>\n<p><code><strong>max_workers=4<\/strong><\/code> \u2014 \u044d\u0442\u043e \u043d\u0430\u0448 \u043d\u0430\u0442\u0443\u0440\u0430\u043b\u044c\u043d\u044b\u0439 rate limit. \u041f\u044f\u0442\u044b\u0439 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043f\u043e\u0434\u043e\u0436\u0434\u0451\u0442 \u0432 \u043e\u0447\u0435\u0440\u0435\u0434\u0438, \u0430 \u043d\u0435 \u043f\u043e\u043b\u043e\u0436\u0438\u0442 \u0441\u0435\u0440\u0432\u0435\u0440.<\/p>\n<\/li>\n<li>\n<p><code><strong>DownloadResult<\/strong><\/code><strong> \u2014 <\/strong><code><strong>frozen<\/strong><\/code><strong> dataclass \u0441\u043e <\/strong><code><strong>slots<\/strong><\/code> \u2014 \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043c\u044b Senior-\u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438, \u0430 \u043d\u0435 \u0430\u043d\u0438\u043c\u0430\u0442\u043e\u0440\u044b \u0433\u0438\u0444\u043e\u043a.<\/p>\n<\/li>\n<li>\n<p><strong>\u0424\u043e\u0440\u043c\u0430\u0442 \u0441\u0435\u043b\u0435\u043a\u0442\u043e\u0440<\/strong> <code>bv*[height&lt;=1080][ext=mp4]+ba[ext=m4a]<\/code>  \u2014 \u0437\u0430\u0441\u043b\u0443\u0436\u0438\u0432\u0430\u0435\u0442 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u043d\u043e \u0432\u043a\u0440\u0430\u0442\u0446\u0435: \u0431\u0435\u0440\u0451\u043c \u043b\u0443\u0447\u0448\u0435\u0435 \u0432\u0438\u0434\u0435\u043e mp4 \u0434\u043e  1080p + \u043b\u0443\u0447\u0448\u0435\u0435 \u0430\u0443\u0434\u0438\u043e m4a \u0438 \u043c\u0451\u0440\u0436\u0438\u043c. Telegram \u043b\u044e\u0431\u0438\u0442 mp4\/H.264\/AAC.<\/p>\n<\/li>\n<\/ul>\n<p>\u0425\u0435\u043d\u0434\u043b\u0435\u0440 \u0442\u0435\u043f\u0435\u0440\u044c \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u043e:<\/p>\n<p>python         <\/p>\n<pre><code class=\"python\">@router.message(F.text.regexp(URL_REGEX))async def handle_url(    message: Message,    downloader: DownloaderService,  # \u0438\u043d\u0436\u0435\u043a\u0442\u0438\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 workflow_data) -&gt; None:    status = await message.answer(\"\u23f3 \u041f\u0440\u0438\u043d\u044f\u043b, \u043a\u0430\u0447\u0430\u044e...\")    try:        result = await downloader.download(message.text)    except DownloadError:        await status.edit_text(\"\u274c \u041d\u0435 \u0441\u043c\u043e\u0433 \u0441\u043a\u0430\u0447\u0430\u0442\u044c. \u041f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u0441\u0441\u044b\u043b\u043a\u0443.\")        return    await message.answer_video(        FSInputFile(result.file_path),        caption=result.title[:1024],    )    await status.delete()    result.file_path.unlink(missing_ok=True)<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412\u0441\u0451. Event Loop \u0431\u043e\u043b\u044c\u0448\u0435 \u043d\u0435 \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u0442\u0441\u044f, \u0431\u043e\u0442 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0432\u0441\u0435\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u043c \u043f\u0430\u0440\u0430\u043b\u043b\u0435\u043b\u044c\u043d\u043e, \u0430 \u0442\u044f\u0436\u0451\u043b\u0443\u044e \u0440\u0430\u0431\u043e\u0442\u0443 \u043c\u043e\u043b\u0447\u0430 \u043f\u0435\u0440\u0435\u043c\u0430\u043b\u044b\u0432\u0430\u044e\u0442 4 \u0440\u0430\u0431\u043e\u0447\u0438\u0445 \u043f\u043e\u0442\u043e\u043a\u0430.<\/p>\n<p>\u041d\u043e \u0435\u0441\u0442\u044c \u043e\u0434\u043d\u0430 \u044d\u0441\u0442\u0435\u0442\u0438\u0447\u0435\u0441\u043a\u0430\u044f \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430.<\/p>\n<hr\/>\n<h3>\u041f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0431\u0430\u0440: \u043a\u0430\u043a \u043f\u0440\u043e\u0431\u0440\u043e\u0441\u0438\u0442\u044c \u043f\u0440\u043e\u0446\u0435\u043d\u0442\u044b \u0438\u0437 \u043f\u043e\u0442\u043e\u043a\u0430 \u0432 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0443<\/h3>\n<p>\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043d\u0435 \u043b\u044e\u0431\u0438\u0442 \u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u043d\u0430 \u0441\u0442\u0430\u0442\u0438\u0447\u043d\u043e\u0435 \u00ab\u043a\u0430\u0447\u0430\u044e&#8230;\u00bb \u043c\u0438\u043d\u0443\u0442\u0443. \u0415\u043c\u0443 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0432\u0438\u0434\u0435\u0442\u044c <strong>45% \u2192 72% \u2192 89%<\/strong>. \u0418 \u0432\u043e\u0442 \u0442\u0443\u0442 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u043c\u0430\u0433\u0438\u044f \u043c\u0435\u0436\u043f\u043e\u0442\u043e\u0447\u043d\u043e\u0439 \u043a\u043e\u043c\u043c\u0443\u043d\u0438\u043a\u0430\u0446\u0438\u0438.<\/p>\n<h4>\u0412 \u0447\u0451\u043c \u043f\u043e\u0434\u0441\u0442\u0430\u0432\u0430<\/h4>\n<p><code>yt-dlp<\/code> \u0434\u0430\u0451\u0442 \u043d\u0430\u043c <code>progress_hooks<\/code> \u2014 <strong>\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439<\/strong> \u043a\u043e\u043b\u043b\u0431\u044d\u043a, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u0451\u0440\u0433\u0430\u0435\u0442\u0441\u044f \u0438\u0437 \u0440\u0430\u0431\u043e\u0447\u0435\u0433\u043e \u043f\u043e\u0442\u043e\u043a\u0430 \u043d\u0430 \u043a\u0430\u0436\u0434\u043e\u043c \u0447\u0430\u043d\u043a\u0435. \u0412\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043e\u043d \u0442\u0430\u043a:<\/p>\n<p>python         <\/p>\n<pre><code class=\"python\">def hook(d: dict) -&gt; None:    if d[\"status\"] == \"downloading\":        print(d[\"_percent_str\"])  # '  45.3%'<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u0430: \u0438\u0437 \u044d\u0442\u043e\u0433\u043e \u043a\u043e\u043b\u043b\u0431\u044d\u043a\u0430 <strong>\u043d\u0435\u043b\u044c\u0437\u044f \u043d\u0430\u043f\u0440\u044f\u043c\u0443\u044e<\/strong> \u0432\u044b\u0437\u0432\u0430\u0442\u044c <code>await message.edit_text(...)<\/code>. \u041c\u044b \u0432 \u0447\u0443\u0436\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435, \u0442\u0430\u043c \u043d\u0435\u0442 Event Loop. \u041f\u043e\u043f\u044b\u0442\u043a\u0430 \u0441\u0434\u0435\u043b\u0430\u0442\u044c <a href=\"http:\/\/asyncio.run\" rel=\"noopener noreferrer nofollow\"><code>asyncio.run<\/code><\/a><code>()<\/code> \u0432\u043d\u0443\u0442\u0440\u0438 \u0445\u0443\u043a\u0430 \u2014 \u0433\u0430\u0440\u0430\u043d\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c RuntimeError \u0438\/\u0438\u043b\u0438 race condition.<\/p>\n<h4>\u0420\u0435\u0448\u0435\u043d\u0438\u0435: asyncio.run_coroutine_threadsafe + throttling<\/h4>\n<p>\u041d\u0430\u043c \u043d\u0443\u0436\u0435\u043d <strong>\u043c\u043e\u0441\u0442<\/strong> \u043c\u0435\u0436\u0434\u0443 \u043f\u043e\u0442\u043e\u043a\u043e\u043c \u0438 \u0446\u0438\u043a\u043b\u043e\u043c. \u0423 <code>asyncio<\/code> \u0435\u0441\u0442\u044c \u0440\u043e\u0432\u043d\u043e \u043e\u0434\u043d\u0430 \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u0434\u043b\u044f \u044d\u0442\u043e\u0433\u043e \u2014 <code>run_coroutine_threadsafe<\/code>. \u041e\u043d\u0430 thread-safe \u043f\u043b\u0430\u043d\u0438\u0440\u0443\u0435\u0442 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0443 \u0432 \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u044b\u0439 Event Loop \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 <code>concurrent.futures.Future<\/code>.<\/p>\n<p>\u0418 \u0441\u0440\u0430\u0437\u0443 \u0432\u0442\u043e\u0440\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442: Telegram <strong>\u043b\u044e\u0442\u043e \u043d\u0435 \u043b\u044e\u0431\u0438\u0442<\/strong>, \u043a\u043e\u0433\u0434\u0430 \u0435\u043c\u0443 \u0441\u043f\u0430\u043c\u044f\u0442 <code>editMessageText<\/code>. \u041b\u0438\u043c\u0438\u0442 \u2014 \u043f\u0440\u0438\u043c\u0435\u0440\u043d\u043e \u043e\u0434\u043d\u043e \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0432 \u0441\u0435\u043a\u0443\u043d\u0434\u0443 \u043d\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435. \u0415\u0441\u043b\u0438 \u0432\u044b \u0431\u0443\u0434\u0435\u0442\u0435 \u0434\u0451\u0440\u0433\u0430\u0442\u044c API \u043a\u0430\u0436\u0434\u044b\u0435 200 \u043c\u0438\u043b\u043b\u0438\u0441\u0435\u043a\u0443\u043d\u0434, \u0441\u043b\u043e\u0432\u0438\u0442\u0435 <code>429 Too Many Requests<\/code> \u0438 \u0444\u043b\u0443\u0434-\u0431\u0430\u043d. \u0417\u043d\u0430\u0447\u0438\u0442, \u043d\u0443\u0436\u0435\u043d <strong>\u0442\u0440\u043e\u0442\u0442\u043b\u0438\u043d\u0433<\/strong>.<\/p>\n<h4>\u041a\u043b\u0430\u0441\u0441 ProgressReporter<\/h4>\n<p>python         <\/p>\n<pre><code class=\"python\">from __future__ import annotationsimport asyncioimport timefrom typing import Anyfrom aiogram.types import Messageclass ProgressReporter:    \"\"\"    \u041c\u043e\u0441\u0442 \u043c\u0435\u0436\u0434\u0443 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u043c \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0445\u0443\u043a\u043e\u043c yt-dlp \u0438 \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u043c    \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435\u043c \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0432 Telegram.    \u041a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0444\u0438\u0447\u0438:      * Throttling (\u043d\u0435 \u0447\u0430\u0449\u0435 \u0440\u0430\u0437 \u0432 N \u0441\u0435\u043a\u0443\u043d\u0434) \u2014 \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u0441\u043b\u043e\u0432\u0438\u0442\u044c 429.      * Threadsafe-\u043f\u043b\u0430\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u043a\u043e\u0440\u0443\u0442\u0438\u043d \u0432 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 Event Loop.      * \u0418\u0433\u043d\u043e\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u043e\u0448\u0438\u0431\u043e\u043a \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f (\u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043c\u043e\u0433\u043b\u043e \u0431\u044b\u0442\u044c \u0443\u0434\u0430\u043b\u0435\u043d\u043e).    \"\"\"    __slots__ = (\"_message\", \"_loop\", \"_min_interval\", \"_last_update\", \"_last_text\")    def __init__(        self,        message: Message,        loop: asyncio.AbstractEventLoop,        min_interval: float = 2.0,    ) -&gt; None:        self._message = message        self._loop = loop        self._min_interval = min_interval        self._last_update: float = 0.0        self._last_text: str = \"\"    def __call__(self, d: dict[str, Any]) -&gt; None:        \"\"\"\u042d\u0442\u043e\u0442 \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0441\u044f yt-dlp \u0438\u0437 \u0440\u0430\u0431\u043e\u0447\u0435\u0433\u043e \u043f\u043e\u0442\u043e\u043a\u0430.\"\"\"        status = d.get(\"status\")        if status == \"downloading\":            text = self._format_downloading(d)        elif status == \"finished\":            text = \"\ud83d\udd27 \u041e\u0431\u0440\u0430\u0431\u043e\u0442\u043a\u0430 (ffmpeg)...\"        else:            return        now = time.monotonic()        if now - self._last_update &lt; self._min_interval:            return        if text == self._last_text:            return        self._last_update = now        self._last_text = text        # \u041f\u043b\u0430\u043d\u0438\u0440\u0443\u0435\u043c \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0443 \u0432 \u0447\u0443\u0436\u043e\u0439 event loop \u2014 thread-safe.        # \u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u043d\u0430\u0441 \u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u0443\u0435\u0442, \u043e\u0448\u0438\u0431\u043a\u0438 \u0433\u043b\u043e\u0442\u0430\u0435\u043c.        asyncio.run_coroutine_threadsafe(self._safe_edit(text), self._loop)    async def _safe_edit(self, text: str) -&gt; None:        try:            await self._message.edit_text(text)        except Exception:            # TelegramBadRequest (message is not modified),            # TelegramRetryAfter, message deleted \u0438 \u043f\u0440\u043e\u0447\u0435\u0435.            # \u041f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0431\u0430\u0440 \u043d\u0435 \u0441\u0442\u043e\u0438\u0442 \u0442\u043e\u0433\u043e, \u0447\u0442\u043e\u0431\u044b \u0440\u043e\u043d\u044f\u0442\u044c \u0437\u0430\u0434\u0430\u0447\u0443.            pass    @staticmethod    def _format_downloading(d: dict[str, Any]) -&gt; str:        total = d.get(\"total_bytes\") or d.get(\"total_bytes_estimate\") or 0        downloaded = d.get(\"downloaded_bytes\") or 0        speed = d.get(\"speed\") or 0        if not total:            return f\"\u2b07\ufe0f \u0421\u043a\u0430\u0447\u0430\u043d\u043e {downloaded \/ 1024 \/ 1024:.1f} MB\"        percent = downloaded \/ total * 100        bar_len = 20        filled = int(bar_len * percent \/ 100)        bar = \"\u2588\" * filled + \"\u2591\" * (bar_len - filled)        speed_mb = speed \/ 1024 \/ 1024 if speed else 0        return (            f\"\u2b07\ufe0f \u0417\u0430\u0433\u0440\u0443\u0437\u043a\u0430\\n\"            f\"&lt;code&gt;[{bar}] {percent:.1f}%&lt;\/code&gt;\\n\"            f\"\u0421\u043a\u043e\u0440\u043e\u0441\u0442\u044c: {speed_mb:.2f} MB\/s\"        )<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h4>\u0421\u0432\u044f\u0437\u044b\u0432\u0430\u0435\u043c \u0432\u0441\u0451 \u0432\u043c\u0435\u0441\u0442\u0435<\/h4>\n<p>\u0425\u0435\u043d\u0434\u043b\u0435\u0440 \u043f\u0440\u0438\u043d\u0438\u043c\u0430\u0435\u0442 \u043e\u043a\u043e\u043d\u0447\u0430\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u0432\u0438\u0434:<\/p>\n<p>python         <\/p>\n<pre><code class=\"python\">@router.message(F.text.regexp(URL_REGEX))async def handle_url(    message: Message,    downloader: DownloaderService,) -&gt; None:    status_msg = await message.answer(\"\u23f3 \u041f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430...\")    loop = asyncio.get_running_loop()    reporter = ProgressReporter(status_msg, loop, min_interval=2.0)    try:        result = await downloader.download(            message.text,            progress_hook=reporter,        )    except DownloadError:        await status_msg.edit_text(\"\u274c \u041d\u0435 \u0443\u0434\u0430\u043b\u043e\u0441\u044c \u0441\u043a\u0430\u0447\u0430\u0442\u044c. \u0421\u0441\u044b\u043b\u043a\u0430 \u0431\u0438\u0442\u0430\u044f \u0438\u043b\u0438 \u043a\u043e\u043d\u0442\u0435\u043d\u0442 \u043f\u0440\u0438\u0432\u0430\u0442\u043d\u044b\u0439.\")        return    except Exception:        logger.exception(\"Unexpected error\")        await status_msg.edit_text(\"\ud83d\udca5 \u0412\u043d\u0443\u0442\u0440\u0435\u043d\u043d\u044f\u044f \u043e\u0448\u0438\u0431\u043a\u0430. \u0423\u0436\u0435 \u0447\u0438\u043d\u044e.\")        return    try:        await message.answer_video(            FSInputFile(result.file_path),            caption=f\"\ud83c\udfac {result.title}\"[:1024],            supports_streaming=True,        )        await status_msg.delete()    finally:        result.file_path.unlink(missing_ok=True)<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0427\u0442\u043e \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438:<\/p>\n<ol>\n<li>\n<p><strong>Event Loop \u0441\u0432\u043e\u0431\u043e\u0434\u0435\u043d<\/strong> \u2014 \u043b\u044e\u0431\u043e\u0435 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u0439 \u043a\u0430\u0447\u0430\u044e\u0442 \u043f\u0430\u0440\u0430\u043b\u043b\u0435\u043b\u044c\u043d\u043e (\u0434\u043e \u043b\u0438\u043c\u0438\u0442\u0430 \u043f\u0443\u043b\u0430).<\/p>\n<\/li>\n<li>\n<p><strong>\u041f\u0440\u043e\u0433\u0440\u0435\u0441\u0441 \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u0442\u0441\u044f \u043f\u043b\u0430\u0432\u043d\u043e<\/strong> \u2014 \u043d\u0435 \u0447\u0430\u0449\u0435 \u0440\u0430\u0437\u0430 \u0432 2 \u0441\u0435\u043a\u0443\u043d\u0434\u044b, \u0447\u0442\u043e \u0432\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u0442\u0441\u044f \u0432 \u043b\u0438\u043c\u0438\u0442\u044b Telegram.<\/p>\n<\/li>\n<li>\n<p><strong>\u041c\u0435\u0436\u043f\u043e\u0442\u043e\u0447\u043d\u0430\u044f \u043a\u043e\u043c\u043c\u0443\u043d\u0438\u043a\u0430\u0446\u0438\u044f \u043a\u043e\u0440\u0440\u0435\u043a\u0442\u043d\u0430<\/strong> \u2014 \u0447\u0435\u0440\u0435\u0437 <code>run_coroutine_threadsafe<\/code>, \u0430 \u043d\u0435 \u0447\u0435\u0440\u0435\u0437 \u0445\u0430\u043a\u0438 \u0441 <a href=\"http:\/\/asyncio.new\" rel=\"noopener noreferrer nofollow\"><code>asyncio.new<\/code><\/a><code>_event_loop()<\/code> \u0432 \u043a\u0430\u0436\u0434\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435.<\/p>\n<\/li>\n<li>\n<p><strong>\u041e\u0448\u0438\u0431\u043a\u0438 \u043d\u0435 \u0440\u043e\u043d\u044f\u044e\u0442 \u0437\u0430\u0434\u0430\u0447\u0443<\/strong> \u2014 \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u043e\u0431\u0451\u0440\u043d\u0443\u0442\u043e \u0432 try\/except, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0431\u0430\u0440 \u2014 \u0432\u0435\u0449\u044c \u043e\u043f\u0446\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u0430\u044f.<\/p>\n<\/li>\n<\/ol>\n<hr\/>\n<h3>\u041f\u043e\u0434\u0432\u043e\u0434\u043d\u044b\u0435 \u043a\u0430\u043c\u043d\u0438, \u043e \u043a\u043e\u0442\u043e\u0440\u044b\u0445 \u0441\u0442\u043e\u0438\u0442 \u0437\u043d\u0430\u0442\u044c<\/h3>\n<p>\u0411\u044b\u0441\u0442\u0440\u044b\u0439 \u0447\u0435\u043a-\u043b\u0438\u0441\u0442, \u0447\u0442\u043e\u0431\u044b \u0432\u044b \u043d\u0435 \u043d\u0430\u0441\u0442\u0443\u043f\u0438\u043b\u0438 \u043d\u0430 \u0433\u0440\u0430\u0431\u043b\u0438 \u043f\u043e\u0441\u043b\u0435 \u0434\u0435\u043f\u043b\u043e\u044f:<\/p>\n<ul>\n<li>\n<p><strong>\u041b\u0438\u043c\u0438\u0442 50 \u041c\u0411 \u0443 \u0431\u043e\u0442\u043e\u0432<\/strong> \u043d\u0430 <code>sendVideo<\/code> \u0447\u0435\u0440\u0435\u0437 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 Bot API. \u0420\u0435\u0448\u0430\u0435\u0442\u0441\u044f \u043b\u0438\u0431\u043e self-hosted Bot API (\u043b\u0438\u043c\u0438\u0442 2 \u0413\u0411), \u043b\u0438\u0431\u043e \u043f\u0440\u0435\u0434\u0432\u0430\u0440\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0439 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u043e\u0439 \u0440\u0430\u0437\u043c\u0435\u0440\u0430 \u0447\u0435\u0440\u0435\u0437 <code>extract_info(download=False)<\/code> \u0438 \u043e\u0442\u043a\u0430\u0437\u043e\u043c \u043d\u0430 \u0431\u043e\u043b\u044c\u0448\u0438\u0445 \u0444\u0430\u0439\u043b\u0430\u0445.<\/p>\n<\/li>\n<li>\n<p><strong>FFmpeg \u043e\u0431\u044f\u0437\u0430\u043d \u0431\u044b\u0442\u044c \u0432 PATH<\/strong>. \u0411\u0435\u0437 \u043d\u0435\u0433\u043e <code>merge_output_format<\/code> \u043d\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442, \u0438 \u0432\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0435 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e <code>.mp4<\/code> \u0438 <code>.m4a<\/code>.<\/p>\n<\/li>\n<li>\n<p><strong>YouTube \u043f\u0435\u0440\u0438\u043e\u0434\u0438\u0447\u0435\u0441\u043a\u0438 \u0442\u0440\u0435\u0431\u0443\u0435\u0442 cookies<\/strong> \u0434\u043b\u044f \u0432\u043e\u0437\u0440\u0430\u0441\u0442\u043d\u044b\u0445 \u0438 \u0440\u0435\u0433\u0438\u043e\u043d\u0430\u043b\u044c\u043d\u043e-\u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u043d\u044b\u0445 \u0432\u0438\u0434\u0435\u043e. \u041e\u043f\u0446\u0438\u044f <code>cookiefile<\/code> \u0432 <code>ydl_opts<\/code> \u0441\u043f\u0430\u0441\u0430\u0435\u0442.<\/p>\n<\/li>\n<li>\n<p><strong>\u041e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u0430\u044f \u043e\u0447\u0438\u0441\u0442\u043a\u0430 \u0444\u0430\u0439\u043b\u043e\u0432<\/strong>. \u0415\u0441\u043b\u0438 \u0434\u0432\u0430 \u044e\u0437\u0435\u0440\u0430 \u043f\u0440\u0438\u0441\u043b\u0430\u043b\u0438 \u043e\u0434\u0438\u043d \u0438 \u0442\u043e\u0442 \u0436\u0435 \u0440\u043e\u043b\u0438\u043a, \u0430 <code>outtmpl<\/code> \u0437\u0430\u0432\u044f\u0437\u0430\u043d \u043d\u0430 <code>%(id)s<\/code>, \u0432\u0442\u043e\u0440\u043e\u0439 \u0432\u043e\u0440\u043a\u0435\u0440 \u043c\u043e\u0436\u0435\u0442 \u0443\u0434\u0430\u043b\u0438\u0442\u044c \u0444\u0430\u0439\u043b \u0440\u0430\u043d\u044c\u0448\u0435, \u0447\u0435\u043c \u043f\u0435\u0440\u0432\u044b\u0439 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u0442. \u0420\u0435\u0448\u0430\u0435\u0442\u0441\u044f \u043b\u0438\u0431\u043e \u0443\u043d\u0438\u043a\u0430\u043b\u044c\u043d\u044b\u043c \u043f\u0440\u0435\u0444\u0438\u043a\u0441\u043e\u043c \u043d\u0430 \u0437\u0430\u0434\u0430\u0447\u0443 (UUID \u0432 <code>outtmpl<\/code>), \u043b\u0438\u0431\u043e \u0444\u0430\u0439\u043b\u043e\u0432\u043e\u0439 \u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u043a\u043e\u0439.<\/p>\n<\/li>\n<li>\n<p><strong>\u0411\u044d\u043a\u043f\u0440\u0435\u0448\u0435\u0440<\/strong>. <code>ThreadPoolExecutor<\/code> \u0438\u043c\u0435\u0435\u0442  \u043d\u0435\u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u043d\u0443\u044e \u043e\u0447\u0435\u0440\u0435\u0434\u044c \u2014 \u0435\u0441\u043b\u0438 \u0432\u0430\u0448 \u0431\u043e\u0442 \u0432\u0434\u0440\u0443\u0433 \u043f\u043e\u043f\u0430\u043b \u0432 \u0442\u043e\u043f, \u0441\u043e\u0442\u043d\u0438 \u0437\u0430\u0434\u0430\u0447  \u0432\u0441\u0442\u0430\u043d\u0443\u0442 \u0432 \u043d\u0435\u0451 \u0438 \u0441\u044a\u0435\u0434\u044f\u0442 \u043f\u0430\u043c\u044f\u0442\u044c. \u041e\u0431\u0435\u0440\u043d\u0438\u0442\u0435 \u0432 \u0441\u0435\u043c\u0430\u0444\u043e\u0440 \u0438\u043b\u0438 \u0432 <code>asyncio.Queue<\/code> \u0441 \u043e\u0433\u0440\u0430\u043d\u0438\u0447\u0435\u043d\u0438\u0435\u043c.<\/p>\n<\/li>\n<li>\n<p><strong>Shutdown<\/strong>. \u041f\u0440\u0438 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0435 \u0431\u043e\u0442\u0430 \u043d\u0435 \u0437\u0430\u0431\u0443\u0434\u044c\u0442\u0435 \u0432\u044b\u0437\u0432\u0430\u0442\u044c <code>await downloader.shutdown()<\/code> \u0447\u0435\u0440\u0435\u0437 <code>dp.shutdown()<\/code>, \u0438\u043d\u0430\u0447\u0435 \u043f\u043e\u0442\u043e\u043a\u0438 \u043f\u043e\u0432\u0438\u0441\u043d\u0443\u0442.<\/p>\n<\/li>\n<\/ul>\n<hr\/>\n<h3>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 \u0438 Proof of Concept<\/h3>\n<p>\u041a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0432\u044b\u0432\u043e\u0434\u044b \u0434\u043b\u044f \u0442\u0435\u0445, \u043a\u0442\u043e \u0434\u043e\u043b\u0438\u0441\u0442\u0430\u043b:<\/p>\n<ul>\n<li>\n<p>\u041b\u044e\u0431\u0443\u044e <strong>\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u0443\u044e \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0443<\/strong> \u0432 async-\u043f\u0440\u043e\u0435\u043a\u0442\u0435 \u043d\u0443\u0436\u043d\u043e \u0438\u0437\u043e\u043b\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432 thread pool. <code>yt-dlp<\/code>, <code>requests<\/code>, <code>psycopg2<\/code>, <code>PIL<\/code> \u2014 \u0432\u0441\u0451 \u044d\u0442\u043e \u044f\u0434\u043e\u0432\u0438\u0442\u043e \u0434\u043b\u044f Event Loop.<\/p>\n<\/li>\n<li>\n<p><a href=\"http:\/\/asyncio.run\" rel=\"noopener noreferrer nofollow\"><code><strong>asyncio.run<\/strong><\/code><\/a><code><strong>_coroutine_threadsafe<\/strong><\/code> \u2014 \u0435\u0434\u0438\u043d\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0439 \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u044b\u0439 \u0441\u043f\u043e\u0441\u043e\u0431 \u0433\u043e\u043d\u044f\u0442\u044c \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u044b \u0438\u0437 \u0447\u0443\u0436\u043e\u0433\u043e \u043f\u043e\u0442\u043e\u043a\u0430. \u041d\u0438\u043a\u0430\u043a\u0438\u0445 <code>new_event_loop<\/code>, \u043d\u0438\u043a\u0430\u043a\u0438\u0445 <a href=\"http:\/\/asyncio.run\" rel=\"noopener noreferrer nofollow\"><code>asyncio.run<\/code><\/a> \u0432\u043d\u0443\u0442\u0440\u0438 \u043a\u043e\u043b\u043b\u0431\u044d\u043a\u0430.<\/p>\n<\/li>\n<li>\n<p><strong>Throttling \u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u0435\u043d<\/strong> \u043d\u0430 \u043b\u044e\u0431\u043e\u0439 \u043a\u043e\u043c\u043c\u0443\u043d\u0438\u043a\u0430\u0446\u0438\u0438 \u0441 Telegram API \u0438\u0437-\u043f\u043e\u0434 \u043f\u0440\u043e\u0433\u0440\u0435\u0441\u0441\u0430. \u0418\u043d\u0430\u0447\u0435 \u2014 \u0444\u043b\u0443\u0434-\u0431\u0430\u043d.<\/p>\n<\/li>\n<li>\n<p><strong>\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043d\u043e \u0440\u0430\u0437\u0434\u0435\u043b\u044f\u0439\u0442\u0435 \u0441\u043b\u043e\u0438<\/strong>: \u0445\u0435\u043d\u0434\u043b\u0435\u0440 \u043d\u0435 \u0434\u043e\u043b\u0436\u0435\u043d \u0437\u043d\u0430\u0442\u044c \u043f\u0440\u043e <code>yt-dlp<\/code>, \u0441\u0435\u0440\u0432\u0438\u0441 \u043d\u0435 \u0434\u043e\u043b\u0436\u0435\u043d \u0437\u043d\u0430\u0442\u044c \u043f\u0440\u043e <code>Message<\/code>. \u041c\u044b \u044d\u0442\u043e\u0433\u043e \u043a\u0430\u0441\u0430\u043b\u0438\u0441\u044c \u0447\u0435\u0440\u0435\u0437 <code>ProgressReporter<\/code> \u043a\u0430\u043a callable.<\/p>\n<\/li>\n<\/ul>\n<p>\u0412\u0441\u0451 \u043e\u043f\u0438\u0441\u0430\u043d\u043d\u043e\u0435 \u0443 \u043c\u0435\u043d\u044f \u043a\u0440\u0443\u0442\u0438\u0442\u0441\u044f \u043d\u0430 \u0441\u043a\u0440\u043e\u043c\u043d\u043e\u0439 VPS, \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0441 \u0434\u0435\u043a\u0430\u0431\u0440\u044f, \u043f\u0435\u0440\u0435\u0436\u0438\u043b\u043e \u0430\u043f\u0434\u0435\u0439\u0442\u044b YouTube, \u0430\u043f\u0434\u0435\u0439\u0442\u044b Telegram \u0438 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043c\u043e\u0438\u0445 DDoS-\u0441\u0430\u043c\u043e\u0433\u043e-\u0441\u0435\u0431\u044f-\u044d\u043a\u0441\u043f\u0435\u0440\u0438\u043c\u0435\u043d\u0442\u043e\u0432. \u0411\u043e\u0442 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439, \u043f\u043e\u043a\u0430 \u0431\u0435\u0437 \u0440\u0435\u043a\u043b\u0430\u043c\u044b \u0438 \u0431\u0435\u0437 \u0442\u043e\u0433\u043e \u0441\u0430\u043c\u043e\u0433\u043e \u00ab\u0432\u0432\u0435\u0434\u0438\u0442\u0435 \u043d\u043e\u043c\u0435\u0440 \u0434\u043b\u044f \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0435\u043d\u0438\u044f\u00bb.<\/p>\n<p><strong>\u041c\u043e\u0436\u0435\u0442\u0435 \u043f\u043e\u0442\u044b\u043a\u0430\u0442\u044c \u0432\u0436\u0438\u0432\u0443\u044e, \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c, \u043a\u0430\u043a \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0438 \u043f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c \u0441\u043a\u043e\u0440\u043e\u0441\u0442\u044c \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u2014 \u043f\u0440\u043e\u0435\u043a\u0442 \u0436\u0438\u0432\u0451\u0442 \u0442\u0443\u0442: <\/strong><a href=\"https:\/\/t.me\/skachaesh_bot\" rel=\"noopener noreferrer nofollow\"><strong>@skachaesh_bot<\/strong><\/a><strong>.<\/strong><\/p>\n<div class=\"floating-image\">\n<figure class=\"float full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/cfa\/6cc\/aed\/cfa6ccaedbf917e311da253b09117f01.jpg\" alt=\"\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0431\u043e\u0442\u0430 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0441 \u044e\u0442\u0443\u0431 \u0432 \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c\" title=\"\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0431\u043e\u0442\u0430 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0441 \u044e\u0442\u0443\u0431 \u0432 \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c\" width=\"833\" height=\"1280\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/cfa\/6cc\/aed\/cfa6ccaedbf917e311da253b09117f01.jpg 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/cfa\/6cc\/aed\/cfa6ccaedbf917e311da253b09117f01.jpg 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption>\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0431\u043e\u0442\u0430 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0441 \u044e\u0442\u0443\u0431 \u0432 \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c<\/figcaption><\/div>\n<\/figure>\n<p>\u0415\u0441\u043b\u0438 \u0441\u0442\u0430\u0442\u044c\u044f \u0437\u0430\u0448\u043b\u0430 \u2014 \u043f\u0438\u0448\u0438\u0442\u0435 \u0432 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445, \u043f\u0440\u043e \u0447\u0442\u043e \u0435\u0449\u0451 \u043f\u043e\u043a\u043e\u043f\u0430\u0442\u044c: \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e \u043f\u0440\u043e self-hosted Telegram Bot API \u0438 \u043e\u0431\u0445\u043e\u0434 \u043b\u0438\u043c\u0438\u0442\u0430 50 \u041c\u0411, \u043f\u0440\u043e FSM \u0438 \u043e\u0447\u0435\u0440\u0435\u0434\u0438 \u0437\u0430\u0434\u0430\u0447 \u043d\u0430 Redis, \u0438\u043b\u0438 \u043f\u0440\u043e \u0442\u043e, \u043a\u0430\u043a \u043f\u0440\u0438\u043a\u0440\u0443\u0442\u0438\u0442\u044c \u0441\u044e\u0434\u0430 Celery \u0438 \u0432 \u043a\u0430\u043a\u043e\u0439 \u043c\u043e\u043c\u0435\u043d\u0442 \u044d\u0442\u043e \u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u0441\u044f \u043e\u0432\u0435\u0440\u0438\u043d\u0436\u0438\u043d\u0438\u0440\u0438\u043d\u0433\u043e\u043c.<\/p>\n<\/div>\n<\/div>\n<p>\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/1026238\/\">https:\/\/habr.com\/ru\/articles\/1026238\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u0431\u043e\u0442 \u0434\u043b\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431, \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c, \u0442\u0438\u043a\u0442\u043e\u043a\u0417\u0430\u0447\u0435\u043c \u0432 2026 \u0433\u043e\u0434\u0443 \u043f\u0438\u0441\u0430\u0442\u044c \u0435\u0449\u0451 \u043e\u0434\u0438\u043d \u0437\u0430\u0433\u0440\u0443\u0437\u0447\u0438\u043a \u0432\u0438\u0434\u0435\u043e\u041a\u0430\u0437\u0430\u043b\u043e\u0441\u044c \u0431\u044b, \u0442\u0435\u043c\u0430 \u0437\u0430\u0435\u0437\u0436\u0435\u043d\u043d\u0430\u044f \u0434\u043e \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u044f \u043f\u043e\u043a\u0440\u044b\u0448\u043a\u0438 \u043a\u0430\u043c\u0430\u0437\u0430: \u00ab\u0441\u0434\u0435\u043b\u0430\u0439 \u0431\u043e\u0442\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043a\u0430\u0447\u0430\u0435\u0442 \u0440\u043e\u043b\u0438\u043a\u0438\u00bb. \u041d\u0430 \u0425\u0430\u0431\u0440\u0435 \u0443\u0436\u0435 \u043b\u0435\u0436\u0438\u0442 \u0434\u0435\u0441\u044f\u0442\u043e\u043a \u0442\u0443\u0442\u043e\u0440\u0438\u0430\u043b\u043e\u0432, \u043d\u0430 GitHub \u2014 \u0441\u043e\u0442\u043d\u0438 \u0444\u043e\u0440\u043a\u043e\u0432. \u041d\u043e \u0435\u0441\u043b\u0438 \u0432\u044b \u0445\u043e\u0442\u044c \u0440\u0430\u0437 \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u043b\u0438 \u0442\u0430\u043a\u043e\u0439 \u0431\u043e\u0442 \u0432 \u043f\u0440\u043e\u0434\u0435 \u2014 \u0432\u044b \u0437\u043d\u0430\u0435\u0442\u0435, \u0447\u0442\u043e 90% \u0438\u0437 \u043d\u0438\u0445 \u043f\u0430\u0434\u0430\u044e\u0442 \u043d\u0430 \u0432\u0442\u043e\u0440\u043e\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435, \u0430 \u043e\u0441\u0442\u0430\u0432\u0448\u0438\u0435\u0441\u044f 10% \u0447\u0435\u0441\u0442\u043d\u043e \u0443\u043c\u0438\u0440\u0430\u044e\u0442 \u043d\u0430 \u0432\u0438\u0434\u0435\u043e \u0434\u043b\u0438\u043d\u043d\u0435\u0435 15 \u043c\u0438\u043d\u0443\u0442.\u042f \u043f\u043e\u043b\u0435\u0437 \u0432 \u043e\u0440\u0433\u0430\u043d\u0438\u0447\u0435\u0441\u043a\u0443\u044e \u0432\u044b\u0434\u0430\u0447\u0443 Google, \u0447\u0442\u043e\u0431\u044b \u043f\u043e\u043d\u044f\u0442\u044c, \u043f\u043e\u0447\u0435\u043c\u0443 \u043b\u044e\u0434\u0438 \u0432\u043e\u043e\u0431\u0449\u0435 \u0438\u0449\u0443\u0442 Telegram-\u0431\u043e\u0442\u043e\u0432, \u0430 \u043d\u0435 \u043f\u043e\u043b\u044c\u0437\u0443\u044e\u0442\u0441\u044f \u0432\u0435\u0431-\u0441\u0435\u0440\u0432\u0438\u0441\u0430\u043c\u0438. \u0418 \u0432\u044b\u0434\u0430\u0447\u0430, \u0447\u0435\u0441\u0442\u043d\u043e \u0433\u043e\u0432\u043e\u0440\u044f, \u0443\u0434\u0440\u0443\u0447\u0430\u0435\u0442.\u041c\u0438\u043d\u0438-\u0430\u043d\u0430\u043b\u0438\u0442\u0438\u043a\u0430 \u043f\u043e\u0438\u0441\u043a\u043e\u0432\u043e\u0439 \u0432\u044b\u0434\u0430\u0447\u0438 (\u0438\u043b\u0438 \u043f\u043e\u0447\u0435\u043c\u0443 \u043b\u044e\u0434\u0438 \u0438\u0434\u0443\u0442 \u0432 Telegram)\u041a\u043e\u0433\u0434\u0430 \u044f \u0438\u0437 \u043b\u044e\u0431\u043e\u043f\u044b\u0442\u0441\u0442\u0432\u0430 \u043f\u043e\u0448\u0451\u043b \u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c, \u0447\u0442\u043e \u043d\u0430 \u0441\u0430\u043c\u043e\u043c \u0434\u0435\u043b\u0435 \u0433\u0443\u0433\u043b\u044f\u0442 \u0436\u0438\u0432\u044b\u0435 \u043b\u044e\u0434\u0438 \u0432\u043e\u043a\u0440\u0443\u0433 \u0442\u0435\u043c\u044b \u00ab\u0441\u043a\u0430\u0447\u0430\u0442\u044c \u0432\u0438\u0434\u0435\u043e \u0441 \u044e\u0442\u0443\u0431\u0430\u00bb, \u043a\u0430\u0440\u0442\u0438\u043d\u0430 \u0441\u043b\u043e\u0436\u0438\u043b\u0430\u0441\u044c \u0434\u043e\u0432\u043e\u043b\u044c\u043d\u043e \u043f\u0440\u0435\u0434\u0441\u043a\u0430\u0437\u0443\u0435\u043c\u0430\u044f. \u0411\u0430\u0437\u043e\u0432\u044b\u0439 \u0437\u0430\u043f\u0440\u043e\u0441 \u2014 \u044d\u0442\u043e \u0431\u0430\u043d\u0430\u043b\u044c\u043d\u043e\u0435 \u00ab\u0441\u043a\u0430\u0447\u0430\u0442\u044c \u0432\u0438\u0434\u0435\u043e \u0441 YouTube \u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0435\u00bb, \u0441 \u0432\u0430\u0440\u0438\u0430\u0446\u0438\u044f\u043c\u0438 \u0440\u0430\u0441\u043a\u043b\u0430\u0434\u043a\u0438 \u0438 \u043e\u043f\u0435\u0447\u0430\u0442\u043e\u043a \u0443\u0440\u043e\u0432\u043d\u044f \u00ab\u044e\u0442\u0443\u0431 com\u00bb. \u0414\u0430\u043b\u044c\u0448\u0435 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u0441\u0435\u0433\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044f \u043f\u043e \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0443: \u043a\u043e\u043c\u0443-\u0442\u043e \u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u043d\u0443\u0436\u043d\u043e 1080p (\u043f\u0440\u0438\u0447\u0451\u043c \u043f\u043e\u043b\u043e\u0432\u0438\u043d\u0430 \u043f\u0438\u0448\u0435\u0442 \u00ab1080\u0440\u00bb \u043a\u0438\u0440\u0438\u043b\u043b\u0438\u0446\u0435\u0439, \u0434\u0430\u0436\u0435 \u043d\u0435 \u0437\u0430\u043c\u0435\u0447\u0430\u044f \u044d\u0442\u043e\u0433\u043e), \u0430 \u043a\u043e\u043c\u0443-\u0442\u043e \u2014 \u0433\u043e\u043b\u044b\u0439 MP4-\u0444\u0430\u0439\u043b \u0431\u0435\u0437 \u043b\u0438\u0448\u043d\u0438\u0445 \u043e\u0431\u0432\u0435\u0441\u043e\u0432.\u0412\u0442\u043e\u0440\u0430\u044f \u043e\u0441\u044c \u2014 \u0441\u043f\u043e\u0441\u043e\u0431 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f. \u0427\u0435\u043b\u043e\u0432\u0435\u043a \u0445\u043e\u0447\u0435\u0442 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e, \u043e\u043d\u043b\u0430\u0439\u043d \u0438 \u0436\u0435\u043b\u0430\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u0435\u0437 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438 (\u0432 2026-\u043c \u0444\u043e\u0440\u043c\u0430 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438 \u043f\u0443\u0433\u0430\u0435\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f \u0441\u0438\u043b\u044c\u043d\u0435\u0435, \u0447\u0435\u043c \u0431\u0430\u043d\u043d\u0435\u0440 \u043e\u043d\u043b\u0430\u0439\u043d-\u043a\u0430\u0437\u0438\u043d\u043e, \u0438 \u044f \u0435\u0433\u043e \u043f\u043e\u043d\u0438\u043c\u0430\u044e). \u0418\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u0434 \u044d\u0442\u043e\u0442 \u0438\u043d\u0442\u0435\u043d\u0442 \u0437\u0430\u0442\u043e\u0447\u0435\u043d\u044b \u043f\u0435\u0440\u0432\u044b\u0435 \u0434\u0435\u0441\u044f\u0442\u044c \u0441\u0442\u0440\u043e\u043a \u0432\u044b\u0434\u0430\u0447\u0438: \u0441\u043d\u0430\u0440\u0443\u0436\u0438 \u2014 \u0430\u043a\u043a\u0443\u0440\u0430\u0442\u043d\u044b\u0439 \u043b\u0435\u043d\u0434\u0438\u043d\u0433 \u0441 \u0431\u043e\u043b\u044c\u0448\u043e\u0439 \u043a\u043d\u043e\u043f\u043a\u043e\u0439, \u0432\u043d\u0443\u0442\u0440\u0438 \u2014 \u0442\u0440\u0438 \u0440\u0435\u0434\u0438\u0440\u0435\u043a\u0442\u0430, \u043f\u043e\u043f\u0430\u043f-\u0431\u043b\u043e\u043a\u0438\u0440\u043e\u0432\u0449\u0438\u043a, \u0438\u043d\u0441\u0442\u0430\u043b\u043b\u0435\u0440 Yandex.Browser Setup.exe \u0438, \u0435\u0441\u043b\u0438 \u043f\u043e\u0432\u0435\u0437\u0451\u0442, \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 \u043c\u0430\u0439\u043d\u0435\u0440 \u0432 \u043f\u043e\u0434\u0430\u0440\u043e\u043a. \u0421\u0443\u0442\u044c \u0432\u0441\u0435\u0445 \u044d\u0442\u0438\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432, \u043a\u0430\u043a \u043d\u0438 \u043a\u0440\u0443\u0442\u0438 \u0438\u0445 \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438, \u0441\u0432\u043e\u0434\u0438\u0442\u0441\u044f \u043a \u043e\u0434\u043d\u043e\u043c\u0443 \u2014 \u043d\u0430\u0436\u0430\u0442\u044c \u043e\u0434\u043d\u0443 \u043a\u043d\u043e\u043f\u043a\u0443 \u0438 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c MP4 \u0432 \u0445\u043e\u0440\u043e\u0448\u0435\u043c \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0435. \u0412\u0441\u0451 \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u043e\u0435 \u2014 \u0443\u0436\u0435 \u043d\u0430\u0448\u0438 \u0441 \u0432\u0430\u043c\u0438 \u0438\u043d\u0436\u0435\u043d\u0435\u0440\u043d\u044b\u0435 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b. \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u043e\u0430\u0442\u0435\u043b\u044c \u043d\u0435 \u0445\u043e\u0447\u0435\u0442:\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u0440\u0435\u043a\u043b\u0430\u043c\u0443 \u043a\u0430\u0437\u0438\u043d\u043e \u00ab\u041f\u0438\u043d-\u0410\u043f\u00bb;\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u00ab\u043f\u043b\u0430\u0433\u0438\u043d \u0434\u043b\u044f \u0431\u0440\u0430\u0443\u0437\u0435\u0440\u0430\u00bb;\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c\u0441\u044f, \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0436\u0434\u0430\u0442\u044c \u043f\u043e\u0447\u0442\u0443 \u0438 \u043f\u0440\u043e\u0445\u043e\u0434\u0438\u0442\u044c \u043a\u0430\u043f\u0447\u0443 \u0441 \u043f\u043e\u0436\u0430\u0440\u043d\u044b\u043c\u0438 \u0433\u0438\u0434\u0440\u0430\u043d\u0442\u0430\u043c\u0438;\u043e\u0431\u044a\u044f\u0441\u043d\u044f\u0442\u044c \u0430\u043d\u0442\u0438\u0432\u0438\u0440\u0443\u0441\u0443, \u0447\u0442\u043e setup_downloader_pro_final2.exe \u2014 \u044d\u0442\u043e \u044f\u043a\u043e\u0431\u044b \u043b\u0435\u0433\u0438\u0442\u0438\u043c\u043d\u043e.\u0418\u043c\u0435\u043d\u043d\u043e \u043f\u043e\u044d\u0442\u043e\u043c\u0443 Telegram-\u0431\u043e\u0442 \u0432\u044b\u0438\u0433\u0440\u044b\u0432\u0430\u0435\u0442 \u0443 \u0432\u0435\u0431-\u0441\u0435\u0440\u0432\u0438\u0441\u043e\u0432 \u043a\u0430\u043a \u043a\u043e\u043d\u0446\u0435\u043f\u0442: \u043d\u0435\u0442 \u0440\u0435\u043a\u043b\u0430\u043c\u044b, \u043d\u0435\u0442 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438, \u043d\u0435\u0442 \u0441\u0442\u0440\u0430\u043d\u043d\u044b\u0445 exe-\u0448\u043d\u0438\u043a\u043e\u0432. \u041f\u0440\u0438\u0441\u043b\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u043b \u0444\u0430\u0439\u043b. \u0412\u0441\u0451.\u041f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u043e\u0434\u043d\u0430: \u043d\u0430\u043f\u0438\u0441\u0430\u0442\u044c \u0442\u0430\u043a\u043e\u0439 \u0431\u043e\u0442 \u043f\u0440\u0430\u0432\u0438\u043b\u044c\u043d\u043e \u2014 \u0441\u043b\u043e\u0436\u043d\u0435\u0435, \u0447\u0435\u043c \u043a\u0430\u0436\u0435\u0442\u0441\u044f. \u041e\u0431 \u044d\u0442\u043e\u043c \u0438 \u0441\u0442\u0430\u0442\u044c\u044f.\u0421\u0442\u0435\u043a: \u043f\u043e\u0447\u0435\u043c\u0443 Python + aiogram 3 + yt-dlp\u041a\u043e\u0440\u043e\u0442\u043a\u043e \u0438 \u043f\u043e \u0434\u0435\u043b\u0443, \u0431\u0435\u0437 \u043e\u0447\u0435\u0440\u0435\u0434\u043d\u043e\u0433\u043e \u0441\u0440\u0430\u0432\u043d\u0435\u043d\u0438\u044f \u0441 Go \u0438 Node.Python 3.11+ \u2014 \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e asyncio \u0432 11-\u0439 \u0432\u0435\u0442\u043a\u0435 \u0441\u0442\u0430\u043b \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0431\u044b\u0441\u0442\u0440\u044b\u043c, TaskGroup \u043f\u0440\u0438\u0435\u0445\u0430\u043b, \u0430 tomllib \u0432\u0441\u0442\u0440\u043e\u0435\u043d.aiogram 3.x \u2014 \u0441\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0439 \u0440\u043e\u0443\u0442\u0435\u0440, Dispatcher \u043d\u0430 DI, FSM \u0438\u0437 \u043a\u043e\u0440\u043e\u0431\u043a\u0438, \u043f\u043e\u043b\u043d\u043e\u0446\u0435\u043d\u043d\u044b\u0439 Type Hinting, middlewares, \u0444\u0438\u043b\u044c\u0442\u0440\u044b. \u042d\u0442\u043e \u043d\u0435 python-telegram-bot \u0432 \u0435\u0433\u043e \u043a\u043b\u0430\u0441\u0441\u0438\u0447\u0435\u0441\u043a\u043e\u043c \u043f\u0435\u0440\u0435\u0433\u0440\u0443\u0436\u0435\u043d\u043d\u043e\u043c \u0432\u0438\u0434\u0435 \u0438 \u043d\u0435 telebot \u043e\u0431\u0440\u0430\u0437\u0446\u0430 2018-\u0433\u043e.yt-dlp \u2014 \u0444\u043e\u0440\u043a youtube-dl, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0436\u0438\u0432\u0435\u0435 \u0432\u0441\u0435\u0445 \u0436\u0438\u0432\u044b\u0445. \u041f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0435\u0442 \u043d\u0435 \u0442\u043e\u043b\u044c\u043a\u043e YouTube, \u043d\u043e \u0438 \u043e\u043a\u043e\u043b\u043e 1500 \u0441\u0430\u0439\u0442\u043e\u0432 (TikTok, VK, Instagram, X\/Twitter, Vimeo, Rutube \u0438 \u0442\u0430\u043a \u0434\u0430\u043b\u0435\u0435). \u041f\u0430\u0440\u0441\u0435\u0440\u044b \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u044e\u0442\u0441\u044f \u0447\u0430\u0449\u0435, \u0447\u0435\u043c \u044f \u043a\u043e\u043c\u043c\u0438\u0447\u0443.\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043d\u043e \u0437\u0430\u0434\u0430\u0447\u0430 \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u0442\u0440\u0438\u0432\u0438\u0430\u043b\u044c\u043d\u043e: \u043f\u0440\u0438\u043d\u044f\u043b \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u2192 \u0432\u044b\u0434\u0440\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u2192 \u0432\u044b\u0437\u0432\u0430\u043b yt-dlp \u2192 \u043e\u0442\u043f\u0440\u0430\u0432\u0438\u043b \u0444\u0430\u0439\u043b. \u0418 \u0432\u043e\u0442 \u0437\u0434\u0435\u0441\u044c \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u0441\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435.\u0413\u043b\u0430\u0432\u043d\u0430\u044f \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430: \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0439 IO \u0438 \u0441\u043c\u0435\u0440\u0442\u044c Event Loop\u041d\u043e\u0432\u0438\u0447\u043e\u043a, \u043d\u0430\u043f\u0438\u0441\u0430\u0432\u0448\u0438\u0439 \u043f\u0435\u0440\u0432\u043e\u0433\u043e \u0431\u043e\u0442\u0430 \u043d\u0430 aiogram, \u0434\u0435\u043b\u0430\u0435\u0442 \u043f\u0440\u0438\u043c\u0435\u0440\u043d\u043e \u0442\u0430\u043a:python         @router.message(F.text.startswith(&#171;http&#187;))async def download_handler(message: Message) -&gt; None:    url = message.text    with YoutubeDL({&#171;outtmpl&#187;: &#171;video.mp4&#187;}) as ydl:        ydl.download([url])  # \ud83d\udd25 \u0442\u0443\u0442 \u0432\u0441\u0451 \u0438 \u0443\u043c\u0438\u0440\u0430\u0435\u0442    await message.answer_video(FSInputFile(&#171;video.mp4&#187;))\u041d\u0430 \u043e\u0434\u043d\u043e\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435 \u044d\u0442\u043e \u0434\u0430\u0436\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442. \u041d\u0430 \u0432\u0442\u043e\u0440\u043e\u043c \u2014 \u0431\u043e\u0442 \u043f\u0440\u0435\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0432 \u0442\u044b\u043a\u0432\u0443.\u041f\u043e\u0447\u0435\u043c\u0443 \u0442\u0430\u043a \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442asyncio \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0432 \u043e\u0434\u043d\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435 \u2014 \u044d\u0442\u043e \u043a\u043e\u043e\u043f\u0435\u0440\u0430\u0442\u0438\u0432\u043d\u0430\u044f \u043c\u043d\u043e\u0433\u043e\u0437\u0430\u0434\u0430\u0447\u043d\u043e\u0441\u0442\u044c. Event Loop \u0436\u043e\u043d\u0433\u043b\u0438\u0440\u0443\u0435\u0442 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0430\u043c\u0438, \u043f\u0435\u0440\u0435\u043a\u043b\u044e\u0447\u0430\u044f\u0441\u044c \u043c\u0435\u0436\u0434\u0443 \u043d\u0438\u043c\u0438 \u043d\u0430 await. \u041f\u043e\u043a\u0430 \u043e\u0434\u043d\u0430 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0430 \u043d\u0435 \u0432\u0441\u0442\u0440\u0435\u0442\u0438\u043b\u0430 await \u0438\u043b\u0438 \u043d\u0435 \u0437\u0430\u0432\u0435\u0440\u0448\u0438\u043b\u0430\u0441\u044c \u2014 \u043e\u0441\u0442\u0430\u043b\u044c\u043d\u044b\u0435 \u0436\u0434\u0443\u0442.yt-dlp \u2014 \u043f\u043e\u043b\u043d\u043e\u0441\u0442\u044c\u044e \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u0430\u044f \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430. \u041f\u043e\u0434 \u043a\u0430\u043f\u043e\u0442\u043e\u043c \u043e\u043d\u0430:\u0425\u043e\u0434\u0438\u0442 HTTP-\u0437\u0430\u043f\u0440\u043e\u0441\u0430\u043c\u0438 \u0447\u0435\u0440\u0435\u0437 urllib (\u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0439 \u0441\u043e\u043a\u0435\u0442).\u041f\u0430\u0440\u0441\u0438\u0442 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b, \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u0442 URL \u0441\u0442\u0440\u0438\u043c\u043e\u0432.\u0421\u043a\u0430\u0447\u0438\u0432\u0430\u0435\u0442 \u0447\u0430\u043d\u043a\u0438 \u0447\u0435\u0440\u0435\u0437 \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0438\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0438 IO.\u0412\u044b\u0437\u044b\u0432\u0430\u0435\u0442 ffmpeg \u0447\u0435\u0440\u0435\u0437 subprocess \u2014 \u044d\u0442\u043e \u0442\u043e\u0436\u0435 \u0431\u043b\u043e\u043a.\u041a\u043e\u0433\u0434\u0430 \u0432\u044b \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u0442\u0435 ydl.download([url]) \u0432 async def-\u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0435, \u0432\u044b \u0431\u0443\u043a\u0432\u0430\u043b\u044c\u043d\u043e \u0437\u0430\u043c\u043e\u0440\u0430\u0436\u0438\u0432\u0430\u0435\u0442\u0435 Event Loop \u043d\u0430 \u0432\u0441\u0451 \u0432\u0440\u0435\u043c\u044f \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f. \u041f\u0440\u0435\u0434\u0441\u0442\u0430\u0432\u044c\u0442\u0435: \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u0410 \u043f\u0440\u0438\u0441\u043b\u0430\u043b \u0441\u0441\u044b\u043b\u043a\u0443 \u043d\u0430 2-\u0447\u0430\u0441\u043e\u0432\u043e\u0439 \u0441\u0442\u0440\u0438\u043c \u0432 4K. \u041d\u0430 2 \u0447\u0430\u0441\u0430 \u0432\u0430\u0448 \u0431\u043e\u0442 \u043f\u0435\u0440\u0435\u0441\u0442\u0430\u0451\u0442 \u043e\u0442\u0432\u0435\u0447\u0430\u0442\u044c \u0430\u0431\u0441\u043e\u043b\u044e\u0442\u043d\u043e \u0432\u0441\u0435\u043c \u2014 \u043a\u043d\u043e\u043f\u043a\u0438 \u043d\u0435 \u043d\u0430\u0436\u0438\u043c\u0430\u044e\u0442\u0441\u044f, \/start \u0438\u0433\u043d\u043e\u0440\u0438\u0440\u0443\u0435\u0442\u0441\u044f, middlewares \u043c\u043e\u043b\u0447\u0430\u0442. \u0418 \u043d\u0435\u0442, GIL \u0437\u0434\u0435\u0441\u044c \u043d\u0435 \u043f\u043e\u043c\u043e\u0436\u0435\u0442: \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u0434\u0430\u0436\u0435 \u043d\u0435 \u0432 \u043d\u0451\u043c, \u0430 \u0432 \u0442\u043e\u043c, \u0447\u0442\u043e \u0443 \u0432\u0430\u0441 \u043e\u0434\u0438\u043d \u043f\u043e\u0442\u043e\u043a \u0434\u043b\u044f \u0432\u0441\u0435\u0439 \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e\u0439 \u043c\u0430\u0448\u0438\u043d\u044b.\u041c\u043d\u043e\u0433\u0438\u0435 \u043d\u0430\u0438\u0432\u043d\u043e \u0434\u0443\u043c\u0430\u044e\u0442: \u00ab\u043d\u0443 \u0442\u0430\u043a GIL \u0436\u0435 \u0432\u0441\u0451 \u0440\u0430\u0432\u043d\u043e \u043e\u0442\u043f\u0443\u0441\u043a\u0430\u0435\u0442\u0441\u044f \u043d\u0430 IO, \u043f\u0440\u043e\u0431\u043b\u0435\u043c \u0431\u044b\u0442\u044c \u043d\u0435 \u0434\u043e\u043b\u0436\u043d\u043e\u00bb. \u041f\u0440\u043e\u0431\u043b\u0435\u043c\u0430 \u0435\u0441\u0442\u044c \u2014 GIL \u043e\u0442\u043f\u0443\u0441\u043a\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u0438\u043d\u0442\u0435\u0440\u043f\u0440\u0435\u0442\u0430\u0442\u043e\u0440\u0430 CPython, \u043d\u043e asyncio \u043f\u0440\u043e \u044d\u0442\u043e \u043d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u0437\u043d\u0430\u0435\u0442. \u0414\u043b\u044f Event Loop \u043b\u044e\u0431\u043e\u0439 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u0432\u044b\u0437\u043e\u0432 \u2014 \u044d\u0442\u043e \u0447\u0451\u0440\u043d\u044b\u0439 \u044f\u0449\u0438\u043a, \u0438\u0437 \u043a\u043e\u0442\u043e\u0440\u043e\u0433\u043e \u043d\u0435 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442\u0441\u044f \u0443\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435.\u0412\u044b\u0432\u043e\u0434: \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u043a\u043e\u0434 \u0432 async-\u0445\u0435\u043d\u0434\u043b\u0435\u0440\u0435 = DoS \u0441\u0430\u043c\u043e\u0433\u043e \u0441\u0435\u0431\u044f.\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043d\u043e\u0435 \u0440\u0435\u0448\u0435\u043d\u0438\u0435: \u0432\u044b\u043d\u043e\u0441\u0438\u043c \u0440\u0430\u0431\u043e\u0442\u0443 \u0432 \u043f\u043e\u0442\u043e\u043a\u0420\u0435\u0448\u0435\u043d\u0438\u0439 \u0443 \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u044b \u0442\u0440\u0438:asyncio.to_thread() \u2014 \u0441\u0430\u0445\u0430\u0440 \u043f\u043e\u0432\u0435\u0440\u0445 run_in_executor \u0441 \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u043c \u043f\u0443\u043b\u043e\u043c. Pythonic, \u043b\u0435\u043d\u0438\u0432, \u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442 \u0434\u043b\u044f 95% \u0441\u043b\u0443\u0447\u0430\u0435\u0432.\u041a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 ThreadPoolExecutor \u2014 \u043a\u043e\u0433\u0434\u0430 \u043d\u0443\u0436\u0435\u043d \u043a\u043e\u043d\u0442\u0440\u043e\u043b\u044c \u043d\u0430\u0434 \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e\u043c \u0432\u043e\u0440\u043a\u0435\u0440\u043e\u0432 \u0438 \u043e\u0447\u0435\u0440\u0435\u0434\u044c\u044e.ProcessPoolExecutor \u2014 \u043a\u043e\u0433\u0434\u0430 \u0435\u0441\u0442\u044c \u0440\u0435\u0430\u043b\u044c\u043d\u044b\u0439 CPU-bound (\u043d\u0430\u043c \u043d\u0435 \u043f\u043e\u0434\u0445\u043e\u0434\u0438\u0442, yt-dlp \u2014 \u044d\u0442\u043e IO-bound).\u042f \u0431\u0435\u0440\u0443 \u0432\u0430\u0440\u0438\u0430\u043d\u0442 \u21162 \u2014 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u0439 \u043f\u0443\u043b. \u041f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u0435\u0441\u043b\u0438 \u0432\u044b \u043f\u0443\u0441\u0442\u0438\u0442\u0435 \u0432\u0441\u0451 \u0432 \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u0439 \u043f\u0443\u043b \u0447\u0435\u0440\u0435\u0437 to_thread, \u0442\u043e \u043f\u0440\u0438 50 \u043f\u0430\u0440\u0430\u043b\u043b\u0435\u043b\u044c\u043d\u044b\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u0430\u0445 \u0443 \u0432\u0430\u0441 \u0443\u043b\u0435\u0442\u0438\u0442 \u0432 \u043d\u0435\u0431\u0435\u0441\u0430 \u0438 \u043f\u0430\u043c\u044f\u0442\u044c, \u0438 \u0441\u0435\u0442\u044c. \u041d\u0443\u0436\u0435\u043d rate limiting \u043d\u0430 \u0443\u0440\u043e\u0432\u043d\u0435 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u044b.\u0421\u0435\u0440\u0432\u0438\u0441\u043d\u044b\u0439 \u0441\u043b\u043e\u0439 DownloaderServicepython         from __future__ import annotationsimport asyncioimport loggingfrom concurrent.futures import ThreadPoolExecutorfrom dataclasses import dataclassfrom pathlib import Pathfrom typing import Any, Callablefrom yt_dlp import YoutubeDLfrom yt_dlp.utils import DownloadErrorlogger = logging.getLogger(__name__)@dataclass(slots=True, frozen=True)class DownloadResult:    file_path: Path    title: str    duration: int    filesize: intclass DownloaderService:    &#171;&#187;&#187;    \u0421\u0435\u0440\u0432\u0438\u0441 \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u044f \u0432\u0438\u0434\u0435\u043e. \u0412\u0441\u044e \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u044e\u0449\u0443\u044e \u0440\u0430\u0431\u043e\u0442\u0443 \u0432\u044b\u043d\u043e\u0441\u0438\u0442    \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u044b\u0439 ThreadPoolExecutor, \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043c\u0435\u0448\u0430\u0442\u044c Event Loop.    &#171;&#187;&#187;    def __init__(        self,        download_dir: Path,        max_workers: int = 4,    ) -&gt; None:        self._download_dir = download_dir        self._download_dir.mkdir(parents=True, exist_ok=True)        # \u041e\u0433\u0440\u0430\u043d\u0438\u0447\u0438\u0432\u0430\u0435\u043c \u043a\u043e\u043b\u0438\u0447\u0435\u0441\u0442\u0432\u043e \u043e\u0434\u043d\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0445 \u0441\u043a\u0430\u0447\u0438\u0432\u0430\u043d\u0438\u0439.        # \u0411\u043e\u043b\u044c\u0448\u0435 4-8 \u0441\u043c\u044b\u0441\u043b\u0430 \u043d\u0435 \u0438\u043c\u0435\u0435\u0442: \u0443\u043f\u0440\u0451\u043c\u0441\u044f \u0432 \u0441\u0435\u0442\u044c\/\u0434\u0438\u0441\u043a\/RAM.        self._executor = ThreadPoolExecutor(            max_workers=max_workers,            thread_name_prefix=&#187;ytdlp-worker&#187;,        )    async def download(        self,        url: str,        progress_hook: Callable[[dict[str, Any]], None] | None = None,    ) -&gt; DownloadResult:        &#171;&#187;&#187;        \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u0430\u044f \u043e\u0431\u0451\u0440\u0442\u043a\u0430. \u0412\u043d\u0443\u0442\u0440\u0438 \u2014 \u0447\u0435\u0441\u0442\u043d\u044b\u0439 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 yt-dlp,        \u043d\u043e \u0438\u0441\u043f\u043e\u043b\u043d\u044f\u0435\u0442\u0441\u044f \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u043c \u043f\u043e\u0442\u043e\u043a\u0435.        &#171;&#187;&#187;        loop = asyncio.get_running_loop()        return await loop.run_in_executor(            self._executor,            self._blocking_download,            url,            progress_hook,        )    def _blocking_download(        self,        url: str,        progress_hook: Callable[[dict[str, Any]], None] | None,    ) -&gt; DownloadResult:        &#171;&#187;&#187;\u042d\u0442\u043e\u0442 \u043c\u0435\u0442\u043e\u0434 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0435\u0442\u0441\u044f \u0412\u041d\u0415 event loop, \u0432 \u0440\u0430\u0431\u043e\u0447\u0435\u043c \u043f\u043e\u0442\u043e\u043a\u0435.&#187;&#187;&#187;        ydl_opts: dict[str, Any] = {            # \u041b\u0443\u0447\u0448\u0435\u0435 mp4 \u0434\u043e 1080p + \u043b\u0443\u0447\u0448\u0438\u0439 m4a, \u0441\u043c\u0451\u0440\u0436\u0435\u043d\u043d\u044b\u0435 \u0432 mp4.            &#171;format&#187;: &#171;bv*[height&lt;=1080][ext=mp4]+ba[ext=m4a]\/b[ext=mp4]\/b&#187;,            &#171;merge_output_format&#187;: &#171;mp4&#187;,            &#171;outtmpl&#187;: str(self._download_dir \/ &#171;%(id)s.%(ext)s&#187;),            &#171;quiet&#187;: True,            &#171;no_warnings&#187;: True,            &#171;noprogress&#187;: True,  # \u0441\u0432\u043e\u0439 \u0445\u0443\u043a, \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 stdout \u043d\u0435 \u043d\u0443\u0436\u0435\u043d            &#171;concurrent_fragment_downloads&#187;: 4,            &#171;retries&#187;: 3,        }        if progress_hook is not None:            ydl_opts[&#171;progress_hooks&#187;] = [progress_hook]        try:            with YoutubeDL(ydl_opts) as ydl:                info = ydl.extract_info(url, download=True)                file_path = Path(ydl.prepare_filename(info))        except DownloadError as e:            logger.warning(&#171;yt-dlp error for %s: %s&#187;, url, e)            raise        return DownloadResult(            file_path=file_path,            title=info.get(&#171;title&#187;, &#171;video&#187;),            duration=int(info.get(&#171;duration&#187;) or 0),            filesize=file_path.stat().st_size,        )    async def shutdown(self) -&gt; None:        self._executor.shutdown(wait=True, cancel_futures=False)\u0427\u0442\u043e \u0437\u0434\u0435\u0441\u044c \u0432\u0430\u0436\u043d\u043e:run_in_executor \u0441 \u043a\u0430\u0441\u0442\u043e\u043c\u043d\u044b\u043c \u043f\u0443\u043b\u043e\u043c \u2014 \u043c\u044b \u043d\u0435 \u0442\u0440\u043e\u0433\u0430\u0435\u043c \u0434\u0435\u0444\u043e\u043b\u0442\u043d\u044b\u0439 executor, \u043e\u043d \u043d\u0443\u0436\u0435\u043d \u0434\u043b\u044f \u0434\u0440\u0443\u0433\u0438\u0445 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439 (to_thread \u0443 aiogram \u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a).max_workers=4 \u2014 \u044d\u0442\u043e \u043d\u0430\u0448 \u043d\u0430\u0442\u0443\u0440\u0430\u043b\u044c\u043d\u044b\u0439 rate limit. \u041f\u044f\u0442\u044b\u0439 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043f\u043e\u0434\u043e\u0436\u0434\u0451\u0442 \u0432 \u043e\u0447\u0435\u0440\u0435\u0434\u0438, \u0430 \u043d\u0435 \u043f\u043e\u043b\u043e\u0436\u0438\u0442 \u0441\u0435\u0440\u0432\u0435\u0440.DownloadResult \u2014 frozen dataclass \u0441\u043e slots \u2014 \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e \u043c\u044b Senior-\u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438, \u0430 \u043d\u0435 \u0430\u043d\u0438\u043c\u0430\u0442\u043e\u0440\u044b \u0433\u0438\u0444\u043e\u043a.\u0424\u043e\u0440\u043c\u0430\u0442 \u0441\u0435\u043b\u0435\u043a\u0442\u043e\u0440 bv*[height&lt;=1080][ext=mp4]+ba[ext=m4a]  \u2014 \u0437\u0430\u0441\u043b\u0443\u0436\u0438\u0432\u0430\u0435\u0442 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u043d\u043e \u0432\u043a\u0440\u0430\u0442\u0446\u0435: \u0431\u0435\u0440\u0451\u043c \u043b\u0443\u0447\u0448\u0435\u0435 \u0432\u0438\u0434\u0435\u043e mp4 \u0434\u043e  1080p + \u043b\u0443\u0447\u0448\u0435\u0435 \u0430\u0443\u0434\u0438\u043e m4a \u0438 \u043c\u0451\u0440\u0436\u0438\u043c. Telegram \u043b\u044e\u0431\u0438\u0442 mp4\/H.264\/AAC.\u0425\u0435\u043d\u0434\u043b\u0435\u0440 \u0442\u0435\u043f\u0435\u0440\u044c \u0432\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u043e:python         @router.message(F.text.regexp(URL_REGEX))async def handle_url(    message: Message,    downloader: DownloaderService,  # \u0438\u043d\u0436\u0435\u043a\u0442\u0438\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 workflow_data) -&gt; None:    status = await message.answer(&#171;\u23f3 \u041f\u0440\u0438\u043d\u044f\u043b, \u043a\u0430\u0447\u0430\u044e&#8230;&#187;)    try:        result = await downloader.download(message.text)    except DownloadError:        await status.edit_text(&#171;\u274c \u041d\u0435 \u0441\u043c\u043e\u0433 \u0441\u043a\u0430\u0447\u0430\u0442\u044c. \u041f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u0441\u0441\u044b\u043b\u043a\u0443.&#187;)        return    await message.answer_video(        FSInputFile(result.file_path),        caption=result.title[:1024],    )    await status.delete()    result.file_path.unlink(missing_ok=True)\u0412\u0441\u0451. Event Loop \u0431\u043e\u043b\u044c\u0448\u0435 \u043d\u0435 \u0431\u043b\u043e\u043a\u0438\u0440\u0443\u0435\u0442\u0441\u044f, \u0431\u043e\u0442 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0432\u0441\u0435\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u043c \u043f\u0430\u0440\u0430\u043b\u043b\u0435\u043b\u044c\u043d\u043e, \u0430 \u0442\u044f\u0436\u0451\u043b\u0443\u044e \u0440\u0430\u0431\u043e\u0442\u0443 \u043c\u043e\u043b\u0447\u0430 \u043f\u0435\u0440\u0435\u043c\u0430\u043b\u044b\u0432\u0430\u044e\u0442 4 \u0440\u0430\u0431\u043e\u0447\u0438\u0445 \u043f\u043e\u0442\u043e\u043a\u0430.\u041d\u043e \u0435\u0441\u0442\u044c \u043e\u0434\u043d\u0430 \u044d\u0441\u0442\u0435\u0442\u0438\u0447\u0435\u0441\u043a\u0430\u044f \u043f\u0440\u043e\u0431\u043b\u0435\u043c\u0430.\u041f\u0440\u043e\u0433\u0440\u0435\u0441\u0441-\u0431\u0430\u0440: \u043a\u0430\u043a \u043f\u0440\u043e\u0431\u0440\u043e\u0441\u0438\u0442\u044c \u043f\u0440\u043e\u0446\u0435\u043d\u0442\u044b \u0438\u0437 \u043f\u043e\u0442\u043e\u043a\u0430 \u0432 \u043a\u043e\u0440\u0443\u0442\u0438\u043d\u0443\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c \u043d\u0435 \u043b\u044e\u0431\u0438\u0442 \u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u043d\u0430 \u0441\u0442\u0430\u0442\u0438\u0447\u043d\u043e\u0435 \u00ab\u043a\u0430\u0447\u0430\u044e&#8230;\u00bb \u043c\u0438\u043d\u0443\u0442\u0443. \u0415\u043c\u0443 \u0445\u043e\u0447\u0435\u0442\u0441\u044f \u0432\u0438\u0434\u0435\u0442\u044c 45% \u2192 72% \u2192 89%. \u0418 \u0432\u043e\u0442 \u0442\u0443\u0442 \u043d\u0430\u0447\u0438\u043d\u0430\u0435\u0442\u0441\u044f \u043c\u0430\u0433\u0438\u044f \u043c\u0435\u0436\u043f\u043e\u0442\u043e\u0447\u043d\u043e\u0439 \u043a\u043e\u043c\u043c\u0443\u043d\u0438\u043a\u0430\u0446\u0438\u0438.\u0412 \u0447\u0451\u043c \u043f\u043e\u0434\u0441\u0442\u0430\u0432\u0430yt-dlp \u0434\u0430\u0451\u0442 \u043d\u0430\u043c progress_hooks \u2014 \u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u043a\u043e\u043b\u043b\u0431\u044d\u043a, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0434\u0451\u0440\u0433\u0430\u0435\u0442\u0441\u044f \u0438\u0437 \u0440\u0430\u0431\u043e\u0447\u0435\u0433\u043e \u043f\u043e\u0442\u043e\u043a\u0430 \u043d\u0430 \u043a\u0430\u0436\u0434\u043e\u043c \u0447\u0430\u043d\u043a\u0435. \u0412\u044b\u0433\u043b\u044f\u0434\u0438\u0442 \u043e\u043d \u0442\u0430\u043a:python         def hook(d: dict)&#8230;<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-476858","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/476858","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=476858"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/476858\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=476858"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=476858"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=476858"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}