{"id":475188,"date":"2025-09-16T09:17:51","date_gmt":"2025-09-16T09:17:51","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=475188"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=475188","title":{"rendered":"\u0410\u0433\u0440\u0435\u0433\u0430\u0446\u0438\u044f \u0438 \u043f\u0430\u0440\u0441\u0438\u043d\u0433 XML RSS \u043b\u0435\u043d\u0442\u044b \u043d\u0430 Python"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u043a\u0430\u043a \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e Python \u0441\u043e\u0431\u0438\u0440\u0430\u0442\u044c \u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u043d\u043e\u0432\u043e\u0441\u0442\u0438 \u0441 \u0441\u0430\u0439\u0442\u0430, \u0438\u043c\u0435\u044e\u0449\u0435\u0433\u043e RSS.<\/p>\n<p>RSS \u2013 \u044d\u0442\u043e \u043f\u0440\u043e\u0441\u0442\u043e\u0439 XML-\u0444\u043e\u0440\u043c\u0430\u0442, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u0441\u043e\u0434\u0435\u0440\u0436\u0430\u0442\u0441\u044f \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0438, \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u044f, \u0441\u0441\u044b\u043b\u043a\u0438 \u0438 \u0434\u0430\u0442\u044b \u043f\u0443\u0431\u043b\u0438\u043a\u0430\u0446\u0438\u0438. \u0421\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u0441\u0430\u0439\u0442\u044b \u0447\u0430\u0441\u0442\u043e \u043f\u0440\u0435\u0434\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u044e\u0442 RSS-\u043b\u0435\u043d\u0442\u044b \u0434\u043b\u044f \u0443\u0434\u043e\u0431\u043d\u043e\u0433\u043e \u0447\u0442\u0435\u043d\u0438\u044f \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0432 \u0441\u0442\u043e\u0440\u043e\u043d\u043d\u0438\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f\u0445. \u0411\u043b\u0430\u0433\u043e\u0434\u0430\u0440\u044f \u044d\u0442\u043e\u043c\u0443 \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043b\u044e\u0431\u043e\u0439 \u044f\u0437\u044b\u043a \u043f\u0440\u043e\u0433\u0440\u0430\u043c\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u00ab\u043f\u043e\u0434\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f\u00bb \u043d\u0430 \u043f\u043e\u0442\u043e\u043a \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u043f\u043e \u0441\u0432\u043e\u0435\u043c\u0443 \u0443\u0441\u043c\u043e\u0442\u0440\u0435\u043d\u0438\u044e.<\/p>\n<p>\u0412 \u043d\u0430\u0448\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043c\u044b \u0441\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u0441\u043a\u0440\u0438\u043f\u0442 \u043d\u0430 Python, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0437\u0430 \u0437\u0430\u0434\u0430\u043d\u043d\u044b\u0439 \u043f\u0435\u0440\u0438\u043e\u0434 (\u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0437\u0430 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430) \u0441\u043e\u0431\u0435\u0440\u0451\u0442 \u0432\u0441\u0435 \u0437\u0430\u043f\u0438\u0441\u0438 \u0438\u0437 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u0438\u0445 \u043b\u0435\u043d\u0442 \u0441\u0430\u0439\u0442\u0430 BBC, \u043e\u0442\u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u0442 \u0438\u0445 \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443 <strong>\u00ab<\/strong>\u0422\u0440\u0430\u043c\u043f<strong>\u00bb<\/strong> \u0438 \u043e\u043f\u0443\u0431\u043b\u0438\u043a\u0443\u0435\u0442 \u0438\u0442\u043e\u0433\u043e\u0432\u044b\u0439 \u043f\u043e\u0434\u0431\u043e\u0440 \u0432 \u043d\u0430\u0448 Telegram-\u043a\u0430\u043d\u0430\u043b. \u0414\u0430\u043b\u0435\u0435 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043a\u043e\u0434, \u0432\u044b \u043b\u0435\u0433\u043a\u043e \u0441\u043c\u043e\u0436\u0435\u0442\u0435 \u0430\u0434\u0430\u043f\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u043f\u043e\u0434 \u043b\u044e\u0431\u0443\u044e \u0434\u0440\u0443\u0433\u0443\u044e \u043b\u0435\u043d\u0442\u0443 \u0438\u043b\u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u0435 \u0441\u043b\u043e\u0432\u043e.<\/p>\n<p>\u0412\u0441\u0451 \u044d\u0442\u043e \u043c\u044b \u0440\u0430\u0437\u0432\u0435\u0440\u043d\u0451\u043c \u0432 \u043e\u0431\u043b\u0430\u043a\u0435 <a href=\"https:\/\/amvera.ru\"><strong>Amvera<\/strong><\/a>, \u0437\u0430\u043f\u043e\u043b\u043d\u0438\u0432 \u0432\u0441\u0435\u0433\u043e \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u043b\u0435\u0439 \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u0438, \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0438\u0432 \u0442\u0440\u0438 \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0432 IDE.<\/p>\n<p>\u0412 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u043a\u043e\u0434\u0430 \u0432\u0441\u0435 \u043f\u0435\u0440\u0441\u043e\u043d\u0430\u043b\u044c\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0434\u043b\u044f \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043a Telegram (<code>API_ID<\/code>, <code>API_HASH<\/code>, <code>BOT_TOKEN<\/code> \u0438 \u0441\u0441\u044b\u043b\u043a\u0438 \u043d\u0430 \u043a\u0430\u043d\u0430\u043b) \u0437\u0430\u043c\u0435\u043d\u0435\u043d\u044b \u043d\u0430 \u0434\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435.<\/p>\n<h3>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438<\/h3>\n<p>\u041f\u0440\u0435\u0436\u0434\u0435 \u0447\u0435\u043c \u0443\u0433\u043b\u0443\u0431\u043b\u044f\u0442\u044c\u0441\u044f \u0432 \u043a\u043e\u0434, \u043f\u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u043c\u0441\u044f \u0441 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043e\u0439 \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u0438 \u043d\u0443\u0436\u043d\u044b\u043c\u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430\u043c\u0438:<\/p>\n<ul>\n<li>\n<p><strong>requests<\/strong><br \/> \u0414\u043b\u044f \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f HTTP-\u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u043a RSS-\u043b\u0435\u043d\u0442\u0430\u043c (\u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0430 XML).<\/p>\n<\/li>\n<li>\n<p><strong>xml.etree.ElementTree<\/strong><br \/> \u0412\u0441\u0442\u0440\u043e\u0435\u043d\u043d\u044b\u0439 \u0432 Python \u043c\u043e\u0434\u0443\u043b\u044c \u0434\u043b\u044f \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML-\u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u043e\u0432.<\/p>\n<\/li>\n<li>\n<p><strong>pytz<\/strong><br \/> \u0420\u0430\u0431\u043e\u0442\u0430 \u0441 \u0447\u0430\u0441\u043e\u0432\u044b\u043c\u0438 \u043f\u043e\u044f\u0441\u0430\u043c\u0438 \u0438 \u043f\u0440\u0438\u0432\u0435\u0434\u0435\u043d\u0438\u0435\u043c \u0434\u0430\u0442 \u043a \u0435\u0434\u0438\u043d\u043e\u043c\u0443 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u0443 (UTC).<\/p>\n<\/li>\n<li>\n<p><strong>email.utils.parsedate_to_datetime<\/strong><br \/> \u041f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0441\u0442\u0440\u043e\u043a\u0438 <code>pubDate<\/code> \u0438\u0437 RSS \u0432 \u043e\u0431\u044a\u0435\u043a\u0442 <code>datetime<\/code>.<\/p>\n<\/li>\n<li>\n<p><strong>telethon<\/strong><br \/> \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u043a\u043b\u0438\u0435\u043d\u0442 \u0434\u043b\u044f \u0440\u0430\u0431\u043e\u0442\u044b \u0441 Telegram API: \u0430\u0432\u0442\u043e\u0440\u0438\u0437\u0430\u0446\u0438\u044f, \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439.<\/p>\n<\/li>\n<li>\n<p><strong>asyncio<\/strong><br \/> \u041e\u0440\u0433\u0430\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0430 \u0434\u043b\u044f \u043f\u043e\u0441\u043b\u0435\u0434\u043e\u0432\u0430\u0442\u0435\u043b\u044c\u043d\u043e\u0439 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u043b\u0435\u043d\u0442 \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0438 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439.<\/p>\n<\/li>\n<li>\n<p><strong>logging<\/strong><br \/> \u041b\u043e\u0433\u0438 \u0432\u0430\u0436\u043d\u044b \u0434\u043b\u044f \u043f\u043e\u043d\u0438\u043c\u0430\u043d\u0438\u044f, \u043d\u0430 \u043a\u0430\u043a\u043e\u043c \u044d\u0442\u0430\u043f\u0435 \u0447\u0442\u043e-\u0442\u043e \u043f\u043e\u0448\u043b\u043e \u043d\u0435 \u0442\u0430\u043a.<\/p>\n<\/li>\n<\/ul>\n<h3>\u041a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u044f \u0438 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435<\/h3>\n<p>\u0412\u0441\u0435 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0440\u0430\u0431\u043e\u0442\u044b \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u0432\u044b\u043d\u0435\u0441\u0435\u043d\u044b \u0432 \u043d\u0430\u0447\u0430\u043b\u043e \u0444\u0430\u0439\u043b\u0430 \u2013 \u044d\u0442\u043e \u0443\u0434\u043e\u0431\u043d\u043e \u0434\u043b\u044f \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0431\u0435\u0437 \u043f\u0440\u0430\u0432\u043a\u0438 \u043b\u043e\u0433\u0438\u043a\u0438:<\/p>\n<pre><code class=\"python\">DATA_FOLDER = '\/data'  # \u041f\u0430\u043f\u043a\u0430 \u0434\u043b\u044f \u0445\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u043e\u0432, \u0435\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u043eRSS_FEEDS = [    'https:\/\/feeds.bbci.co.uk\/news\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/world\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/business\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/technology\/rss.xml',]# \u041f\u0435\u0440\u0438\u043e\u0434 \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u0438: \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430PERIOD = timedelta(hours=4)TIMEZONE = pytz.utc  # Telegram (\u0432 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0444\u0438\u043a\u0442\u0438\u0432\u043d\u044b\u0435 \u2013 \u0437\u0430\u043c\u0435\u043d\u0438\u0442\u0435 \u043d\u0430 \u0441\u0432\u043e\u0438)API_ID = int(os.getenv('API_ID', 1234567))API_HASH = os.getenv('API_HASH', '\u0432\u0430\u0448_api_hash')BOT_TOKEN = os.getenv('BOT_TOKEN', '\u0432\u0430\u0448_bot_token')CHANNEL_LINK = os.getenv('CHANNEL_LINK', 'https:\/\/t.me\/\u0432\u0430\u0448\\\\_\u043a\u0430\u043d\u0430\u043b')BOT_SESSION_STRING = os.getenv('BOT_SESSION_STRING')  # session string \u0434\u043b\u044f Telethon# HTTP-\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0438, \u0447\u0442\u043e\u0431\u044b \u0441\u0435\u0440\u0432\u0435\u0440 \u0434\u0443\u043c\u0430\u043b, \u0447\u0442\u043e \u043a \u043d\u0435\u043c\u0443 \u0437\u0430\u0445\u043e\u0434\u0438\u0442 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 \u0431\u0440\u0430\u0443\u0437\u0435\u0440HEADERS = {'User-Agent': 'Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)'}<\/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><strong>DATA_FOLDER<\/strong>\u00a0\u2014 \u043a\u0443\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0442\u044c \u0441\u043a\u0430\u0447\u0430\u043d\u043d\u044b\u0439 XML \u0438\u043b\u0438\u00a0\u043b\u043e\u0433\u0438 (\u043f\u0440\u0438\u00a0\u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u0440\u0430\u0441\u0448\u0438\u0440\u0438\u0442\u044c \u0444\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b).<\/p>\n<p><strong>RSS_FEEDS<\/strong>\u00a0\u2014 \u0441\u043f\u0438\u0441\u043e\u043a URL\u2011\u0430\u0434\u0440\u0435\u0441\u043e\u0432 \u0444\u0438\u0434\u043e\u0432 BBC. \u0412\u044b \u043c\u043e\u0436\u0435\u0442\u0435 \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0432\u043e\u0438 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0435 \u0438\u043b\u0438\u00a0\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0443\u0436\u043d\u044b\u0435.<\/p>\n<p><strong>PERIOD<\/strong> \u0438 <strong>TIMEZONE<\/strong>\u00a0\u2014 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u044e\u0442, \u0437\u0430\u00a0\u043a\u0430\u043a\u043e\u0439 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u043a \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043c\u044b \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u043d\u043e\u0432\u043e\u0441\u0442\u0438 \u0438 \u0432\u00a0\u043a\u0430\u043a\u043e\u043c \u0447\u0430\u0441\u043e\u0432\u043e\u043c \u043f\u043e\u044f\u0441\u0435 \u0438\u0445 \u0441\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u0435\u043c.<\/p>\n<p>\u041f\u0435\u0440\u0435\u0434 \u0437\u0430\u043f\u0443\u0441\u043a\u043e\u043c \u043d\u0430\u00a0Amvera (\u0438\u043b\u0438 \u0432\u00a0\u043b\u044e\u0431\u043e\u043c \u0434\u0440\u0443\u0433\u043e\u043c \u043c\u0435\u0441\u0442\u0435) \u043d\u0443\u0436\u043d\u043e \u043d\u0430\u0441\u0442\u0440\u043e\u0438\u0442\u044c \u0440\u0435\u0430\u043b\u044c\u043d\u044b\u0435 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0447\u0435\u0440\u0435\u0437 \u0432\u0435\u0431-\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0438\u043b\u0438 CLI Amvera.<\/p>\n<h3>\u041b\u043e\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435<\/h3>\n<p>\u0427\u0442\u043e\u0431\u044b \u043f\u043e\u043d\u0438\u043c\u0430\u0442\u044c, \u0447\u0442\u043e \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432\u043d\u0443\u0442\u0440\u0438 \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u043d\u0430 \u0443\u0434\u0430\u043b\u0451\u043d\u043d\u043e\u043c \u0441\u0435\u0440\u0432\u0435\u0440\u0435, \u043c\u044b \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0430\u0435\u043c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 \u043c\u043e\u0434\u0443\u043b\u044c <code>logging<\/code>. \u042d\u0442\u043e \u043f\u043e\u0437\u0432\u043e\u043b\u0438\u0442 \u043d\u0435 \u00ab\u0442\u043e\u043d\u0443\u0442\u044c\u00bb \u0432 \u0441\u0442\u0435\u043d\u0435 \u0432\u044b\u0432\u043e\u0434\u0430 \u2013 \u0432\u0441\u0435 \u0432\u0430\u0436\u043d\u044b\u0435 \u0441\u043e\u0431\u044b\u0442\u0438\u044f \u0431\u0443\u0434\u0443\u0442 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u044b \u0432 \u043b\u043e\u0433\u0438.<\/p>\n<pre><code class=\"python\">import logginglogging.basicConfig(    level=logging.INFO,    format='%(asctime)s - %(levelname)s - %(message)s')logger = logging.getLogger('bbc_news_scraper')<\/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<ul>\n<li>\n<p><code>level=<\/code><a href=\"http:\/\/logging.INFO\"><code>logging.INFO<\/code><\/a> \u2014 \u0431\u0443\u0434\u0435\u043c \u043f\u043e\u043b\u0443\u0447\u0430\u0442\u044c \u0432\u0441\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0443\u0440\u043e\u0432\u043d\u044f <code>INFO<\/code> \u0438 \u0432\u044b\u0448\u0435 (<code>WARNING<\/code>, <code>ERROR<\/code>).<\/p>\n<\/li>\n<li>\n<p><code>format='%(asctime)s - %(levelname)s - %(message)s'<\/code> \u2014 \u0432 \u043a\u0430\u0436\u0434\u043e\u0439 \u0441\u0442\u0440\u043e\u043a\u0435 \u043b\u043e\u0433\u0430 \u0431\u0443\u0434\u0435\u0442 \u0432\u0440\u0435\u043c\u044f, \u0443\u0440\u043e\u0432\u0435\u043d\u044c \u0438 \u0442\u0435\u043a\u0441\u0442 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f.<\/p>\n<\/li>\n<li>\n<p><code>logger = logging.getLogger('bbc_news_scraper')<\/code> \u2014 \u0438\u043c\u0435\u043d\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u043b\u043e\u0433\u0433\u0435\u0440, \u0447\u0442\u043e\u0431\u044b \u043f\u0440\u0438 \u043c\u0430\u0441\u0448\u0442\u0430\u0431\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0438 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u043c\u043e\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0440\u0430\u0437\u0434\u0435\u043b\u044f\u0442\u044c \u043b\u043e\u0433\u0438 \u043f\u043e \u043c\u043e\u0434\u0443\u043b\u044f\u043c.<\/p>\n<\/li>\n<\/ul>\n<h3>\u0417\u0430\u0433\u0440\u0443\u0437\u043a\u0430 RSS-\u043b\u0438\u0441\u0442\u0430<\/h3>\n<p>\u041e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0430 \u043f\u0430\u0440\u0441\u0435\u0440\u0430\u00a0\u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c XML \u043f\u043e URL-\u0443 RSS. \u0417\u0434\u0435\u0441\u044c \u043d\u0430 \u043f\u043e\u043c\u043e\u0449\u044c \u043f\u0440\u0438\u0445\u043e\u0434\u0438\u0442 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430 <code>requests<\/code>:<\/p>\n<pre><code class=\"python\">import requestsHEADERS = {'User-Agent': 'Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)'}def fetch_feed_xml(url):    \"\"\"    \u0417\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 RSS-\u0444\u0438\u0434 \u043f\u043e HTTP \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0441\u044b\u0440\u043e\u0435 \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u043c\u043e\u0435 XML.    \u0412 \u0441\u043b\u0443\u0447\u0430\u0435 \u043e\u0448\u0438\u0431\u043a\u0438 \u043b\u043e\u0433\u0438\u0440\u0443\u0435\u043c \u0435\u0435 \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c None.    \"\"\"    try:        resp = requests.get(url, headers=HEADERS, timeout=10)        resp.raise_for_status()  # \u0431\u0440\u043e\u0441\u0438\u0442 \u0438\u0441\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435, \u0435\u0441\u043b\u0438 \u0441\u0442\u0430\u0442\u0443\u0441 \u2260200        return resp.content    except requests.RequestException as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f {url}: {e}\")        return None<\/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<ul>\n<li>\n<p><code>timeout=10<\/code> \u0437\u0430\u0449\u0438\u0449\u0430\u0435\u0442 \u043e\u0442 \u00ab\u0437\u0430\u0432\u0438\u0441\u0448\u0438\u0445\u00bb \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432.<\/p>\n<\/li>\n<li>\n<p><code>resp.raise_for_status()<\/code> \u0441\u0440\u0430\u0437\u0443 \u00ab\u043f\u043e\u0439\u043c\u0430\u0435\u0442\u00bb HTTP-\u043e\u0448\u0438\u0431\u043a\u0438 (404, 500 \u0438 \u0442.\u200a\u0434.).<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u0438 \u043b\u044e\u0431\u043e\u043c \u0438\u0441\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0438 \u043c\u044b \u043b\u043e\u0433\u0438\u0440\u0443\u0435\u043c \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0447\u0435\u0440\u0435\u0437 <code>logger.error<\/code>, \u0447\u0442\u043e\u0431\u044b \u0432 \u043b\u043e\u0433\u0430\u0445 \u0431\u044b\u043b\u043e \u0432\u0438\u0434\u043d\u043e, \u043a\u0430\u043a\u043e\u0439 \u0438\u043c\u0435\u043d\u043d\u043e \u0444\u0438\u0434 \u043d\u0435 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442.<\/p>\n<h3>\u041f\u0430\u0440\u0441\u0438\u043d\u0433 RSS \u0438 \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u044f \u043f\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443<\/h3>\n<p>\u041f\u043e\u0441\u043b\u0435 \u0442\u043e\u0433\u043e \u043a\u0430\u043a \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438 \u00ab\u0441\u044b\u0440\u043e\u0435\u00bb \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u043c\u043e\u0435 RSS, \u043d\u0443\u0436\u043d\u043e \u0438\u0437 \u043d\u0435\u0433\u043e \u0432\u044b\u0442\u0430\u0449\u0438\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0442\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043d\u0430\u043c \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u044b\u00a0\u2014 \u0442\u043e \u0435\u0441\u0442\u044c \u0441\u0442\u0430\u0442\u044c\u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0445 4\u00a0\u0447\u0430\u0441\u043e\u0432 \u0441\u043e \u0441\u043b\u043e\u0432\u043e\u043c <strong>\u00ab\u0422\u0440\u0430\u043c\u043f\u00bb<\/strong> \u0432\u00a0\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0435. \u0414\u043b\u044f\u00a0\u044d\u0442\u043e\u0433\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <code>xml.etree.ElementTree<\/code> \u0438 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u0443\u044e \u0443\u0442\u0438\u043b\u0438\u0442\u0443 \u0434\u043b\u044f \u0440\u0430\u0437\u0431\u043e\u0440\u0430 \u0434\u0430\u0442.<\/p>\n<pre><code class=\"python\">import xml.etree.ElementTree as ETfrom email.utils import parsedate_to_datetimedef parse_feed_items(xml_content: bytes, cutoff: datetime, keyword: str) -&amp;gt; list[dict]:    \"\"\"    \u0420\u0430\u0437\u0431\u0438\u0440\u0430\u0435\u0442 XML, \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u0442  \u0438 \u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u0442 \u043f\u043e \u0434\u0430\u0442\u0435 \u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443.    \u0412\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0441\u043f\u0438\u0441\u043e\u043a \u0441\u043b\u043e\u0432\u0430\u0440\u0435\u0439 {'title', 'link', 'published'}.    \"\"\"    items = []    try:        root = ET.fromstring(xml_content)    except ET.ParseError as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML: {e}\")        return items    # \u0418\u0449\u0435\u043c \u0432\u0441\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b     for item in root.findall('.\/\/item'):        title_el = item.find('title')        link_el = item.find('link')        pubdate_el = item.find('pubDate')        if not (title_el and link_el and pubdate_el):            continue        # \u041f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u0443\u0435\u043c pubDate \u2192 datetime \u0441 \u0447\u0430\u0441\u043e\u0432\u044b\u043c \u043f\u043e\u044f\u0441\u043e\u043c UTC        try:            pub_dt = parsedate_to_datetime(pubdate_el.text)            pub_dt = pub_dt.astimezone(TIMEZONE)        except Exception as e:            logger.warning(f\"\u041d\u0435 \u0443\u0434\u0430\u043b\u043e\u0441\u044c \u0440\u0430\u0441\u043f\u0430\u0440\u0441\u0438\u0442\u044c \u0434\u0430\u0442\u0443: {e}\")            continue        # \u041e\u0442\u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u043c \u043f\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0438 \u043f\u043e \u0441\u043b\u043e\u0432\u0443 \u201c\u0422\u0440\u0430\u043c\u043f\u201d        if pub_dt &amp;gt;= cutoff and keyword.lower() in title_el.text.lower():            items.append({                'title': title_el.text.strip(),                'link': link_el.text.strip(),                'published': pub_dt.strftime('%Y-%m-%d %H:%M:%S %Z')            })    return items<\/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>\u041c\u044b \u0438\u0449\u0435\u043c \u0432\u0441\u0435 &#171; \u0432\u043d\u0443\u0442\u0440\u0438 RSS \u0438 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0443 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0435\u0441\u0442\u044c \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043e\u043a, \u0441\u0441\u044b\u043b\u043a\u0430 \u0438 \u0434\u0430\u0442\u0430.<\/p>\n<p>\u0421 \u043f\u043e\u043c\u043e\u0449\u044c\u044e <code>parsedate_to_datetime<\/code> \u0438\u0437 <code>email.utils<\/code> \u043a\u043e\u043d\u0432\u0435\u0440\u0442\u0438\u0440\u0443\u0435\u043c \u0441\u0442\u0440\u043e\u043a\u0443 \u0432\u0438\u0434\u0430 <code>Fri, 07 Aug 2025 12:34:56 GMT<\/code> \u0432 <code>datetime<\/code> \u0441 \u043f\u043e\u044f\u0441\u043e\u043c UTC.<\/p>\n<p>\u0421\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u0435\u043c \u0434\u0430\u0442\u0443 \u043f\u0443\u0431\u043b\u0438\u043a\u0430\u0446\u0438\u0438 \u0441 <code>cutoff<\/code> (\u0442\u0435\u043a\u0443\u0449\u0438\u043c \u0432\u0440\u0435\u043c\u0435\u043d\u0435\u043c \u043c\u0438\u043d\u0443\u0441 4 \u0447).<\/p>\n<p>\u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0435\u0443\u0441\u043b\u043e\u0432\u0438\u0435\u00a0\u2014 \u0432\u00a0\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0435 \u0434\u043e\u043b\u0436\u043d\u043e \u0432\u0441\u0442\u0440\u0435\u0447\u0430\u0442\u044c\u0441\u044f \u0441\u043b\u043e\u0432\u043e <strong>\u00ab\u0422\u0440\u0430\u043c\u043f\u00bb<\/strong> (\u0431\u0435\u0437 \u0443\u0447\u0435\u0442\u0430 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430).<\/p>\n<p>\u0422\u0430\u043a \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0435 \u043d\u043e\u0432\u043e\u0441\u0442\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0441\u0432\u0435\u0436\u0438\u0435 \u0438 \u0440\u0435\u043b\u0435\u0432\u0430\u043d\u0442\u043d\u044b\u0435 \u043d\u0430\u0448\u0435\u0439 \u0442\u0435\u043c\u0435.<\/p>\n<h3>\u041e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439 \u0432 Telegram<\/h3>\n<p>\u041a\u043e\u0433\u0434\u0430 \u0441\u043f\u0438\u0441\u043e\u043a \u0441\u0432\u0435\u0436\u0438\u0445 \u0441\u0442\u0430\u0442\u0435\u0439 \u0433\u043e\u0442\u043e\u0432, \u0444\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0435\u0433\u043e \u0432 \u0432\u0430\u0448 \u043a\u0430\u043d\u0430\u043b \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e Telethon:<\/p>\n<pre><code class=\"python\">from telethon import TelegramClientfrom telethon.sessions import StringSessionasync def send_to_telegram_channel(message: str):    \"\"\"    \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442 \u0442\u0435\u043a\u0441\u0442 \u0432 Telegram-\u043a\u0430\u043d\u0430\u043b.    \u0412\u0441\u0435 \u043b\u0438\u0447\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 (API_ID, API_HASH, BOT_TOKEN, SESSION) \u0432 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u0437\u0430\u043c\u0435\u043d\u0435\u043d\u044b.    \"\"\"    try:        bot_client = TelegramClient(            StringSession(BOT_SESSION_STRING) if BOT_SESSION_STRING else 'session_bot',            API_ID, API_HASH        )        await bot_client.start(bot_token=BOT_TOKEN)        entity = await bot_client.get_entity(CHANNEL_LINK)        # link_preview=False \u2013 \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043f\u043e\u0434\u0442\u044f\u0433\u0438\u0432\u0430\u043b\u0438\u0441\u044c \u0431\u043e\u043b\u044c\u0448\u0438\u0435 \u043a\u0430\u0440\u0442\u0438\u043d\u043a\u0438        await bot_client.send_message(entity, message, link_preview=False)        logger.info(\"\u0421\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0443\u0441\u043f\u0435\u0448\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u043e \u0432 Telegram\")    except Exception as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u0440\u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0435 \u0432 Telegram: {e}\")    finally:        if bot_client.is_connected():            await bot_client.disconnect()<\/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<ul>\n<li>\n<p><code>StringSession<\/code> \u0438\u043b\u0438 \u043e\u0431\u044b\u0447\u043d\u0430\u044f \u0441\u0435\u0441\u0441\u0438\u044f\u00a0\u2014 \u0445\u0440\u0430\u043d\u0438\u0442 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \u043e \u0432\u0430\u0448\u0435\u043c \u0431\u043e\u0442\u0435.<\/p>\n<\/li>\n<li>\n<p><code>start(bot_token=...)<\/code> \u0434\u0435\u043b\u0430\u0435\u0442 \u0430\u0432\u0442\u043e\u0440\u0438\u0437\u0430\u0446\u0438\u044e \u043f\u043e \u0442\u043e\u043a\u0435\u043d\u0443.<\/p>\n<\/li>\n<li>\n<p><code>get_entity<\/code> \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043d\u0430\u0445\u043e\u0434\u0438\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 \u043a\u0430\u043d\u0430\u043b \u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0435.<\/p>\n<\/li>\n<\/ul>\n<h3>\u041f\u043e\u043b\u043d\u044b\u0439 \u043a\u043e\u0434 (\u0441\u043a\u0435\u043b\u0435\u0442 \u0441\u043a\u0440\u0438\u043f\u0442\u0430)<\/h3>\n<pre><code class=\"python\">import osimport requestsimport xml.etree.ElementTree as ETfrom datetime import datetime, timedeltaimport pytzfrom email.utils import parsedate_to_datetimeimport loggingimport asynciofrom telethon import TelegramClientfrom telethon.sessions import StringSessionimport time# ========== CONFIGURATION ==========DATA_FOLDER = '\/data'RSS_FEEDS = [    'https:\/\/feeds.bbci.co.uk\/news\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/world\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/business\/rss.xml',    'https:\/\/feeds.bbci.co.uk\/news\/technology\/rss.xml',]# Telegram configurationAPI_ID = int(os.getenv('API_ID', 2122521))API_HASH = os.getenv('API_HASH', 'd27621d9925562342baefc013e')BOT_TOKEN = os.getenv('BOT_TOKEN', '759406329:32522CPhmU')CHANNEL_LINK = os.getenv('CHANNEL_LINK', 'https:\/\/t.me\/+vR323w55y')BOT_SESSION_STRING = os.getenv('BOT_SESSION_STRING')HEADERS = {'User-Agent': 'Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)'}TIMEZONE = pytz.utcPERIOD = timedelta(hours=4) # \u041f\u0435\u0440\u0438\u043e\u0434 - 4 \u0447\u0430\u0441\u0430MOSCOW_TZ = pytz.timezone('Europe\/Moscow')# ========== LOGGING ==========logging.basicConfig(    level=logging.INFO,    format='%(asctime)s - %(levelname)s - %(message)s')logger = logging.getLogger('bbc_news_scraper')# ========== UTILS ==========def ensure_dir(path):    os.makedirs(path, exist_ok=True)# ========== FETCH FEED ==========def fetch_feed_xml(url):    try:        resp = requests.get(url, headers=HEADERS, timeout=10)        resp.raise_for_status()        return resp.content    except requests.RequestException as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f {url}: {e}\")        return None# ========== PARSE FEED ITEMS ==========def parse_feed_items(xml_content, cutoff):    items = []    try:        root = ET.fromstring(xml_content)    except ET.ParseError as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML: {e}\")        return items    for item in root.findall('.\/\/item'):        title_el = item.find('title')        link_el = item.find('link')        pubdate_el = item.find('pubDate')        if title_el is None or link_el is None or pubdate_el is None:            continue        try:            pub_dt = parsedate_to_datetime(pubdate_el.text)            pub_dt = pub_dt.astimezone(TIMEZONE)        except Exception:            continue        if pub_dt &amp;gt;= cutoff:            items.append({                'title': title_el.text.strip(),                'link': link_el.text.strip(),                'published': pub_dt.strftime('%Y-%m-%d %H:%M:%S %Z')            })    return items# ========== TELEGRAM SENDER ==========async def send_to_telegram_channel(message):    \"\"\"\u041e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0432 \u0442\u0435\u043b\u0435\u0433\u0440\u0430\u043c-\u043a\u0430\u043d\u0430\u043b\"\"\"    try:        bot_client = TelegramClient(            StringSession(BOT_SESSION_STRING) if BOT_SESSION_STRING else 'session_bot',            API_ID, API_HASH        )        await bot_client.start(bot_token=BOT_TOKEN)        entity = await bot_client.get_entity(CHANNEL_LINK)        await bot_client.send_message(entity, message, link_preview=False)        logger.info(\"\u0421\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u043e \u0432 Telegram\")    except Exception as e:        logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0438 \u0432 Telegram: {e}\")    finally:        if bot_client.is_connected():            await bot_client.disconnect()# ========== MAIN ==========async def main():    logger.info(\"\u0417\u0430\u043f\u0443\u0441\u043a \u0441\u0431\u043e\u0440\u0430 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 BBC \u0437\u0430 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430\")    now = datetime.now(TIMEZONE)    cutoff = now - PERIOD    news_items = []    # \u0421\u0431\u043e\u0440 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0438\u0437 \u0432\u0441\u0435\u0445 RSS-\u043b\u0435\u043d\u0442    for url in RSS_FEEDS:        xml = fetch_feed_xml(url)        if not xml:            continue        parsed = parse_feed_items(xml, cutoff)        logger.info(f\"{url}: \u043d\u0430\u0439\u0434\u0435\u043d\u043e {len(parsed)} \u0441\u0442\u0430\u0442\u0435\u0439 \u0437\u0430 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430\")        news_items.extend(parsed)    # \u0424\u043e\u0440\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0434\u043b\u044f Telegram    if news_items:        message = \"\ud83d\udcf0 *\u041f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 \u043d\u043e\u0432\u043e\u0441\u0442\u0438 BBC \u0437\u0430 4 \u0447\u0430\u0441\u0430:*\"        for item in news_items:            message += f\"\u2022 [{item['title']}]({item['link']})\"        # \u041e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0432 Telegram        await send_to_telegram_channel(message)        logger.info(f\"\u041e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u043e {len(news_items)} \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0432 Telegram\")    else:        logger.info(\"\u041d\u0435\u0442 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0434\u043b\u044f \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0438\")        await send_to_telegram_channel(\"\u2139\ufe0f \u0417\u0430 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u043e\u0442 BBC \u043d\u0435 \u043e\u0431\u043d\u0430\u0440\u0443\u0436\u0435\u043d\u043e\")def calculate_next_run():    \"\"\"\u0412\u044b\u0447\u0438\u0441\u043b\u044f\u0435\u0442 \u0432\u0440\u0435\u043c\u044f \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u0433\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430 (00:00, 04:00, 08:00 \u0438 \u0442.\u0434. \u043f\u043e \u041c\u043e\u0441\u043a\u0432\u0435)\"\"\"    now = datetime.now(MOSCOW_TZ)    current_hour = now.hour    # \u0412\u044b\u0447\u0438\u0441\u043b\u044f\u0435\u043c \u0431\u043b\u0438\u0436\u0430\u0439\u0448\u0438\u0439 \u0447\u0430\u0441, \u043a\u0440\u0430\u0442\u043d\u044b\u0439 4    target_hour = ((current_hour \/\/ 4) * 4 + 4) % 24    next_run = now.replace(hour=target_hour, minute=0, second=0, microsecond=0)    # \u0415\u0441\u043b\u0438 \u0431\u043b\u0438\u0436\u0430\u0439\u0448\u0435\u0435 \u0432\u0440\u0435\u043c\u044f \u0443\u0436\u0435 \u043f\u0440\u043e\u0448\u043b\u043e \u0441\u0435\u0433\u043e\u0434\u043d\u044f, \u043f\u043b\u0430\u043d\u0438\u0440\u0443\u0435\u043c \u043d\u0430 \u0437\u0430\u0432\u0442\u0440\u0430    if next_run &amp;lt; now:        next_run += timedelta(days=1)    logger.info(f\"\u0421\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0439 \u0437\u0430\u043f\u0443\u0441\u043a \u0432: {next_run.strftime('%Y-%m-%d %H:%M:%S %Z%z')}\")    return next_runasync def scheduler():    \"\"\"\u041f\u043b\u0430\u043d\u0438\u0440\u043e\u0432\u0449\u0438\u043a, \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u044e\u0449\u0438\u0439 \u0437\u0430\u0434\u0430\u0447\u0443 \u043a\u0430\u0436\u0434\u044b\u0435 4 \u0447\u0430\u0441\u0430\"\"\"    while True:        next_run = calculate_next_run()        sleep_seconds = (next_run - datetime.now(MOSCOW_TZ)).total_seconds()        logger.info(f\"\u041e\u0436\u0438\u0434\u0430\u043d\u0438\u0435 {sleep_seconds \/ 3600:.2f} \u0447\u0430\u0441\u043e\u0432 \u0434\u043e \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u0433\u043e \u0437\u0430\u043f\u0443\u0441\u043a\u0430\")        await asyncio.sleep(sleep_seconds)        try:            await main()        except Exception as e:            logger.error(f\"\u041e\u0448\u0438\u0431\u043a\u0430 \u0432 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u043c \u0437\u0430\u0434\u0430\u043d\u0438\u0438: {e}\")if __name__ == '__main__':    asyncio.run(scheduler())<\/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<h3>\u0417\u0430\u043f\u0443\u0441\u043a \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u0432 Amvera<\/h3>\n<p>\u0421\u0430\u043c\u043e\u0435 \u0432\u0440\u0435\u043c\u044f \u0440\u0430\u0437\u0432\u0435\u0440\u043d\u0443\u0442\u044c \u043a\u043e\u0434 \u043d\u0430 \u0443\u0434\u0430\u043b\u0435\u043d\u043d\u043e\u043c \u0441\u0435\u0440\u0432\u0435\u0440\u0435. \u0421\u043a\u0440\u0438\u043f\u0442 \u043c\u044b \u0437\u0430\u043f\u0443\u0441\u0442\u0438\u043c \u0432 Amvera, <strong>\u044d\u0442\u043e \u0434\u0430\u0441\u0442 \u043d\u0430\u043c \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043f\u0440\u043e\u0441\u0442\u044b\u0445 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u0439 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u043d\u0430 \u0441\u0435\u0440\u0432\u0435\u0440\u0435 \u0447\u0435\u0440\u0435\u0437 \u0432\u0441\u0442\u0440\u043e\u0435\u043d\u043d\u044b\u0439 CI\/CD \u043e\u0434\u043d\u043e\u0439 \u043a\u043e\u043c\u0430\u043d\u0434\u043e\u0439, <\/strong>\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e\u0435 \u043b\u043e\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0441 \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u043c \u043f\u043e\u0438\u0441\u043a\u043e\u043c \u0438 \u0432\u043e\u0437\u043c\u043e\u0436\u043d\u043e\u0441\u0442\u044c \u043d\u0435 \u043f\u043b\u0430\u0442\u0438\u0442\u044c \u0437\u0430 \u043e\u0441\u0442\u0430\u043d\u043e\u0432\u043b\u0435\u043d\u043d\u044b\u0435 \u043f\u0440\u043e\u0435\u043a\u0442\u044b.<\/p>\n<h4>\u0420\u0430\u0437\u0432\u0435\u0440\u0442\u044b\u0432\u0430\u043d\u0438\u0435<\/h4>\n<ol>\n<li>\n<p>\u0420\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u044f \u0438 \u043f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430 \u0430\u043a\u043a\u0430\u0443\u043d\u0442\u0430: \u043f\u0435\u0440\u0435\u0439\u0434\u0438\u0442\u0435 <a href=\"https:\/\/amvera.ru\">\u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0435<\/a>, \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u0443\u0439\u0442\u0435\u0441\u044c \u0438 \u043f\u043e\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u0442\u0435 \u043f\u043e\u0447\u0442\u0443. \u041f\u0440\u0438 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u0438 \u043e\u0431\u044b\u0447\u043d\u043e \u0431\u043e\u043d\u0443\u0441\u044b \u043d\u0430 \u0431\u0430\u043b\u0430\u043d\u0441 \u0432 \u0440\u0430\u0437\u043c\u0435\u0440\u0435 111 \u0440\u0443\u0431. \u0418\u0445 \u043d\u0430\u043c \u0445\u0432\u0430\u0442\u0438\u0442 \u043d\u0430 \u0442\u0435\u0441\u0442.<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u043d\u043e\u0432\u043e\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430: \u0432 \u043a\u043e\u043d\u0441\u043e\u043b\u0438 Amvera\/\u041f\u0440\u043e\u0435\u043a\u0442\u044b\/\u041d\u043e\u0432\u044b\u0439 \u043f\u0440\u043e\u0435\u043a\u0442. \u0422\u0438\u043f \u0441\u0435\u0440\u0432\u0438\u0441\u0430: <strong>\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435<\/strong>.<\/p>\n<\/li>\n<li>\n<p>\u0417\u0430\u0433\u0440\u0443\u0437\u043a\u0430 \u043a\u043e\u0434\u0430 \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0439 \u043f\u0440\u043e\u0435\u043a\u0442\u0430: \u043e\u0442\u043a\u0440\u043e\u0439\u0442\u0435 \u0432\u043a\u043b\u0430\u0434\u043a\u0443 \u0420\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0439 \u0438 \u0437\u0430\u0433\u0440\u0443\u0437\u0438\u0442\u0435 <a href=\"http:\/\/main.py\"><code>main.py<\/code><\/a> (\u0438\u043b\u0438 \u0432\u0430\u0448 \u043e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u0441\u043a\u0440\u0438\u043f\u0442) \u0438 <code>requirements.txt<\/code>.<\/p>\n<\/li>\n<\/ol>\n<p><code>requirements.txt<\/code> \u043f\u0440\u0438\u043c\u0435\u0440:<\/p>\n<pre><code class=\"json\">requests==2.32.4pytz==2025.7.post1telethon==1.35.0python-dotenv==1.1.1<\/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>\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430 \u0441\u0431\u043e\u0440\u043a\u0438 \u0438 \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0437\u0430\u043f\u0443\u0441\u043a\u0430<\/h4>\n<ul>\n<li>\n<p>\u0412 \u041a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u0438 \u0443\u043a\u0430\u0436\u0438\u0442\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435 <code>Python<\/code>.<\/p>\n<\/li>\n<li>\n<p>\u0412 \u043f\u043e\u043b\u0435 <code>scriptName<\/code> \u0432\u044b\u0431\u0435\u0440\u0438\u0442\u0435 <a href=\"http:\/\/main.py\"><code>main.py<\/code><\/a> (\u0438\u043b\u0438 \u0438\u043c\u044f \u0432\u0430\u0448\u0435\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430).<\/p>\n<\/li>\n<li>\n<p>\u0421\u043e\u0437\u0434\u0430\u0439\u0442\u0435 \u043f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f (<code>API_ID<\/code>, <code>API_HASH<\/code>, <code>BOT_TOKEN<\/code>, <code>CHANNEL_LINK<\/code>, <code>BOT_SESSION_STRING<\/code> \u0438 \u0442.\u0434.) \u0447\u0435\u0440\u0435\u0437 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 Amvera.<\/p>\n<\/li>\n<\/ul>\n<h4>\u0421\u0431\u043e\u0440\u043a\u0430 \u0438 \u0437\u0430\u043f\u0443\u0441\u043a<\/h4>\n<ul>\n<li>\n<p>\u041d\u0430\u0436\u043c\u0438\u0442\u0435 \u00ab\u0421\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c\u00bb \u0438 \u0437\u0430\u0442\u0435\u043c \u00ab\u0421\u043e\u0431\u0440\u0430\u0442\u044c \u043f\u0440\u043e\u0435\u043a\u0442\u00bb.<\/p>\n<\/li>\n<\/ul>\n<figure class=\"full-width \"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/829\/255\/7c9\/8292557c966d3e4ccd72d40e060bd83d.png\" alt=\"\u041f\u0440\u0438\u043c\u0435\u0440 \u043b\u043e\u0433\u0430 \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u0441\u0435\u0440\u0432\u0438\u0441\u0430\" title=\"\u041f\u0440\u0438\u043c\u0435\u0440 \u043b\u043e\u0433\u0430 \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u0441\u0435\u0440\u0432\u0438\u0441\u0430\" width=\"624\" height=\"193\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/829\/255\/7c9\/8292557c966d3e4ccd72d40e060bd83d.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/829\/255\/7c9\/8292557c966d3e4ccd72d40e060bd83d.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption>\u041f\u0440\u0438\u043c\u0435\u0440 \u043b\u043e\u0433\u0430 \u0437\u0430\u043f\u0443\u0449\u0435\u043d\u043d\u043e\u0433\u043e \u0441\u0435\u0440\u0432\u0438\u0441\u0430<\/figcaption><\/div>\n<\/figure>\n<ul>\n<li>\n<p>\u041f\u043e\u0441\u043b\u0435 \u0443\u0441\u043f\u0435\u0448\u043d\u043e\u0439 \u0441\u0431\u043e\u0440\u043a\u0438 Amvera \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u0437\u0430\u043f\u0443\u0441\u0442\u0438\u0442 \u0432\u0430\u0448 \u0441\u043a\u0440\u0438\u043f\u0442. \u041f\u0440\u043e\u0432\u0435\u0440\u0438\u0442\u044c \u043b\u043e\u0433\u0438 \u043c\u043e\u0436\u043d\u043e \u0432 \u0440\u0430\u0437\u0434\u0435\u043b\u0435 <code>\u041b\u043e\u0433\u0438<\/code>.<\/p>\n<\/li>\n<\/ul>\n<figure class=\"\"><img decoding=\"async\" src=\"https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/2cb\/4d7\/5b0\/2cb4d75b0c94abb1397a43adb76c52b4.png\" alt=\"\u041f\u0440\u0438\u043c\u0435\u0440 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c\u044b\u0445 \u043f\u043e RSS \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439\" title=\"\u041f\u0440\u0438\u043c\u0435\u0440 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c\u044b\u0445 \u043f\u043e RSS \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439\" width=\"417\" height=\"305\" sizes=\"auto, (max-width: 780px) 100vw, 50vw\" srcset=\"https:\/\/habrastorage.org\/r\/w780\/getpro\/habr\/upload_files\/2cb\/4d7\/5b0\/2cb4d75b0c94abb1397a43adb76c52b4.png 780w,&#10;       https:\/\/habrastorage.org\/r\/w1560\/getpro\/habr\/upload_files\/2cb\/4d7\/5b0\/2cb4d75b0c94abb1397a43adb76c52b4.png 781w\" loading=\"lazy\" decode=\"async\"\/><\/p>\n<div><figcaption>\u041f\u0440\u0438\u043c\u0435\u0440 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c\u044b\u0445 \u043f\u043e RSS \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439<\/figcaption><\/div>\n<\/figure>\n<h3>\u0417\u0430\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435<\/h3>\n<p>\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043c\u044b \u043f\u043e\u043a\u0430\u0437\u0430\u043b\u0438, \u043a\u0430\u043a \u043b\u0435\u0433\u043a\u043e \u0438 \u0431\u044b\u0441\u0442\u0440\u043e \u043d\u0430\u0441\u0442\u0440\u043e\u0438\u0442\u044c \u0441\u0431\u043e\u0440 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0438\u0437 RSS-\u043b\u0435\u043d\u0442 BBC \u0441 \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u0435\u0439 \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443 \u0438 \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0443\u044e \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0443 \u0432 Telegram-\u043a\u0430\u043d\u0430\u043b.<\/p>\n<p>\u041d\u0430\u0434\u0435\u044e\u0441\u044c, \u044d\u0442\u043e\u0442 \u043f\u0440\u0438\u043c\u0435\u0440 \u0432\u0434\u043e\u0445\u043d\u043e\u0432\u0438\u0442 \u0432\u0430\u0441 \u043d\u0430 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0435 \u043f\u0440\u043e\u0435\u043a\u0442\u044b \u043f\u043e \u043c\u043e\u043d\u0438\u0442\u043e\u0440\u0438\u043d\u0433\u0443 \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0438\u043b\u0438 \u043b\u044e\u0431\u043e\u0433\u043e \u0434\u0440\u0443\u0433\u043e\u0433\u043e \u043a\u043e\u043d\u0442\u0435\u043d\u0442\u0430 \u0432 \u0441\u0435\u0442\u0438.<\/p>\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\/947318\/\">https:\/\/habr.com\/ru\/articles\/947318\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c, \u043a\u0430\u043a \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e Python \u0441\u043e\u0431\u0438\u0440\u0430\u0442\u044c \u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u043d\u043e\u0432\u043e\u0441\u0442\u0438 \u0441 \u0441\u0430\u0439\u0442\u0430, \u0438\u043c\u0435\u044e\u0449\u0435\u0433\u043e RSS.RSS \u2013 \u044d\u0442\u043e \u043f\u0440\u043e\u0441\u0442\u043e\u0439 XML-\u0444\u043e\u0440\u043c\u0430\u0442, \u0432 \u043a\u043e\u0442\u043e\u0440\u043e\u043c \u0441\u043e\u0434\u0435\u0440\u0436\u0430\u0442\u0441\u044f \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0438, \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u044f, \u0441\u0441\u044b\u043b\u043a\u0438 \u0438 \u0434\u0430\u0442\u044b \u043f\u0443\u0431\u043b\u0438\u043a\u0430\u0446\u0438\u0438. \u0421\u043e\u0432\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u0441\u0430\u0439\u0442\u044b \u0447\u0430\u0441\u0442\u043e \u043f\u0440\u0435\u0434\u043e\u0441\u0442\u0430\u0432\u043b\u044f\u044e\u0442 RSS-\u043b\u0435\u043d\u0442\u044b \u0434\u043b\u044f \u0443\u0434\u043e\u0431\u043d\u043e\u0433\u043e \u0447\u0442\u0435\u043d\u0438\u044f \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0432 \u0441\u0442\u043e\u0440\u043e\u043d\u043d\u0438\u0445 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f\u0445. \u0411\u043b\u0430\u0433\u043e\u0434\u0430\u0440\u044f \u044d\u0442\u043e\u043c\u0443 \u043f\u0440\u0430\u043a\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043b\u044e\u0431\u043e\u0439 \u044f\u0437\u044b\u043a \u043f\u0440\u043e\u0433\u0440\u0430\u043c\u043c\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u043f\u043e\u0437\u0432\u043e\u043b\u044f\u0435\u0442 \u00ab\u043f\u043e\u0434\u043f\u0438\u0441\u0430\u0442\u044c\u0441\u044f\u00bb \u043d\u0430 \u043f\u043e\u0442\u043e\u043a \u043d\u043e\u0432\u043e\u0441\u0442\u0435\u0439 \u0438 \u043e\u0431\u0440\u0430\u0431\u0430\u0442\u044b\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u043f\u043e \u0441\u0432\u043e\u0435\u043c\u0443 \u0443\u0441\u043c\u043e\u0442\u0440\u0435\u043d\u0438\u044e.\u0412 \u043d\u0430\u0448\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043c\u044b \u0441\u043e\u0437\u0434\u0430\u0434\u0438\u043c \u0441\u043a\u0440\u0438\u043f\u0442 \u043d\u0430 Python, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0437\u0430 \u0437\u0430\u0434\u0430\u043d\u043d\u044b\u0439 \u043f\u0435\u0440\u0438\u043e\u0434 (\u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u0437\u0430 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430) \u0441\u043e\u0431\u0435\u0440\u0451\u0442 \u0432\u0441\u0435 \u0437\u0430\u043f\u0438\u0441\u0438 \u0438\u0437 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u0438\u0445 \u043b\u0435\u043d\u0442 \u0441\u0430\u0439\u0442\u0430 BBC, \u043e\u0442\u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u0442 \u0438\u0445 \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443 \u00ab\u0422\u0440\u0430\u043c\u043f\u00bb \u0438 \u043e\u043f\u0443\u0431\u043b\u0438\u043a\u0443\u0435\u0442 \u0438\u0442\u043e\u0433\u043e\u0432\u044b\u0439 \u043f\u043e\u0434\u0431\u043e\u0440 \u0432 \u043d\u0430\u0448 Telegram-\u043a\u0430\u043d\u0430\u043b. \u0414\u0430\u043b\u0435\u0435 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043a\u043e\u0434, \u0432\u044b \u043b\u0435\u0433\u043a\u043e \u0441\u043c\u043e\u0436\u0435\u0442\u0435 \u0430\u0434\u0430\u043f\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0435\u0433\u043e \u043f\u043e\u0434 \u043b\u044e\u0431\u0443\u044e \u0434\u0440\u0443\u0433\u0443\u044e \u043b\u0435\u043d\u0442\u0443 \u0438\u043b\u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u0435 \u0441\u043b\u043e\u0432\u043e.\u0412\u0441\u0451 \u044d\u0442\u043e \u043c\u044b \u0440\u0430\u0437\u0432\u0435\u0440\u043d\u0451\u043c \u0432 \u043e\u0431\u043b\u0430\u043a\u0435 Amvera, \u0437\u0430\u043f\u043e\u043b\u043d\u0438\u0432 \u0432\u0441\u0435\u0433\u043e \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u043b\u0435\u0439 \u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u0438, \u0438 \u0432\u044b\u043f\u043e\u043b\u043d\u0438\u0432 \u0442\u0440\u0438 \u043a\u043e\u043c\u0430\u043d\u0434\u044b \u0432 IDE.\u0412 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u043a\u043e\u0434\u0430 \u0432\u0441\u0435 \u043f\u0435\u0440\u0441\u043e\u043d\u0430\u043b\u044c\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0434\u043b\u044f \u0434\u043e\u0441\u0442\u0443\u043f\u0430 \u043a Telegram (API_ID, API_HASH, BOT_TOKEN \u0438 \u0441\u0441\u044b\u043b\u043a\u0438 \u043d\u0430 \u043a\u0430\u043d\u0430\u043b) \u0437\u0430\u043c\u0435\u043d\u0435\u043d\u044b \u043d\u0430 \u0434\u0435\u043c\u043e\u043d\u0441\u0442\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435.\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0438\u041f\u0440\u0435\u0436\u0434\u0435 \u0447\u0435\u043c \u0443\u0433\u043b\u0443\u0431\u043b\u044f\u0442\u044c\u0441\u044f \u0432 \u043a\u043e\u0434, \u043f\u043e\u0437\u043d\u0430\u043a\u043e\u043c\u0438\u043c\u0441\u044f \u0441 \u0430\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u043e\u0439 \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u0438 \u043d\u0443\u0436\u043d\u044b\u043c\u0438 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430\u043c\u0438:requests \u0414\u043b\u044f \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f HTTP-\u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u043a RSS-\u043b\u0435\u043d\u0442\u0430\u043c (\u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0430 XML).xml.etree.ElementTree \u0412\u0441\u0442\u0440\u043e\u0435\u043d\u043d\u044b\u0439 \u0432 Python \u043c\u043e\u0434\u0443\u043b\u044c \u0434\u043b\u044f \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML-\u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u043e\u0432.pytz \u0420\u0430\u0431\u043e\u0442\u0430 \u0441 \u0447\u0430\u0441\u043e\u0432\u044b\u043c\u0438 \u043f\u043e\u044f\u0441\u0430\u043c\u0438 \u0438 \u043f\u0440\u0438\u0432\u0435\u0434\u0435\u043d\u0438\u0435\u043c \u0434\u0430\u0442 \u043a \u0435\u0434\u0438\u043d\u043e\u043c\u0443 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u0443 (UTC).email.utils.parsedate_to_datetime \u041f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0441\u0442\u0440\u043e\u043a\u0438 pubDate \u0438\u0437 RSS \u0432 \u043e\u0431\u044a\u0435\u043a\u0442 datetime.telethon \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u044b\u0439 \u043a\u043b\u0438\u0435\u043d\u0442 \u0434\u043b\u044f \u0440\u0430\u0431\u043e\u0442\u044b \u0441 Telegram API: \u0430\u0432\u0442\u043e\u0440\u0438\u0437\u0430\u0446\u0438\u044f, \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439.asyncio \u041e\u0440\u0433\u0430\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u0430\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0430 \u0434\u043b\u044f \u043f\u043e\u0441\u043b\u0435\u0434\u043e\u0432\u0430\u0442\u0435\u043b\u044c\u043d\u043e\u0439 \u0437\u0430\u0433\u0440\u0443\u0437\u043a\u0438 \u043b\u0435\u043d\u0442 \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0438 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439.logging \u041b\u043e\u0433\u0438 \u0432\u0430\u0436\u043d\u044b \u0434\u043b\u044f \u043f\u043e\u043d\u0438\u043c\u0430\u043d\u0438\u044f, \u043d\u0430 \u043a\u0430\u043a\u043e\u043c \u044d\u0442\u0430\u043f\u0435 \u0447\u0442\u043e-\u0442\u043e \u043f\u043e\u0448\u043b\u043e \u043d\u0435 \u0442\u0430\u043a.\u041a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u044f \u0438 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u0435\u0412\u0441\u0435 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u044b \u0440\u0430\u0431\u043e\u0442\u044b \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u0432\u044b\u043d\u0435\u0441\u0435\u043d\u044b \u0432 \u043d\u0430\u0447\u0430\u043b\u043e \u0444\u0430\u0439\u043b\u0430 \u2013 \u044d\u0442\u043e \u0443\u0434\u043e\u0431\u043d\u043e \u0434\u043b\u044f \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0431\u0435\u0437 \u043f\u0440\u0430\u0432\u043a\u0438 \u043b\u043e\u0433\u0438\u043a\u0438:DATA_FOLDER = &#8216;\/data&#8217;  # \u041f\u0430\u043f\u043a\u0430 \u0434\u043b\u044f \u0445\u0440\u0430\u043d\u0435\u043d\u0438\u044f \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u043e\u0432, \u0435\u0441\u043b\u0438 \u043d\u0443\u0436\u043d\u043eRSS_FEEDS = [    &#8216;https:\/\/feeds.bbci.co.uk\/news\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/world\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/business\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/technology\/rss.xml&#8217;,]# \u041f\u0435\u0440\u0438\u043e\u0434 \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u0438: \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0435 4 \u0447\u0430\u0441\u0430PERIOD = timedelta(hours=4)TIMEZONE = pytz.utc  # Telegram (\u0432 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 \u0444\u0438\u043a\u0442\u0438\u0432\u043d\u044b\u0435 \u2013 \u0437\u0430\u043c\u0435\u043d\u0438\u0442\u0435 \u043d\u0430 \u0441\u0432\u043e\u0438)API_ID = int(os.getenv(&#8216;API_ID&#8217;, 1234567))API_HASH = os.getenv(&#8216;API_HASH&#8217;, &#8216;\u0432\u0430\u0448_api_hash&#8217;)BOT_TOKEN = os.getenv(&#8216;BOT_TOKEN&#8217;, &#8216;\u0432\u0430\u0448_bot_token&#8217;)CHANNEL_LINK = os.getenv(&#8216;CHANNEL_LINK&#8217;, &#8216;https:\/\/t.me\/\u0432\u0430\u0448\\\\_\u043a\u0430\u043d\u0430\u043b&#8217;)BOT_SESSION_STRING = os.getenv(&#8216;BOT_SESSION_STRING&#8217;)  # session string \u0434\u043b\u044f Telethon# HTTP-\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0438, \u0447\u0442\u043e\u0431\u044b \u0441\u0435\u0440\u0432\u0435\u0440 \u0434\u0443\u043c\u0430\u043b, \u0447\u0442\u043e \u043a \u043d\u0435\u043c\u0443 \u0437\u0430\u0445\u043e\u0434\u0438\u0442 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 \u0431\u0440\u0430\u0443\u0437\u0435\u0440HEADERS = {&#8216;User-Agent&#8217;: &#8216;Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)&#8217;}DATA_FOLDER\u00a0\u2014 \u043a\u0443\u0434\u0430 \u043c\u043e\u0436\u043d\u043e \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0442\u044c \u0441\u043a\u0430\u0447\u0430\u043d\u043d\u044b\u0439 XML \u0438\u043b\u0438\u00a0\u043b\u043e\u0433\u0438 (\u043f\u0440\u0438\u00a0\u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u0440\u0430\u0441\u0448\u0438\u0440\u0438\u0442\u044c \u0444\u0443\u043d\u043a\u0446\u0438\u043e\u043d\u0430\u043b).RSS_FEEDS\u00a0\u2014 \u0441\u043f\u0438\u0441\u043e\u043a URL\u2011\u0430\u0434\u0440\u0435\u0441\u043e\u0432 \u0444\u0438\u0434\u043e\u0432 BBC. \u0412\u044b \u043c\u043e\u0436\u0435\u0442\u0435 \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0432\u043e\u0438 \u0441\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u044b\u0435 \u0438\u043b\u0438\u00a0\u043e\u0441\u0442\u0430\u0432\u0438\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u043d\u0443\u0436\u043d\u044b\u0435.PERIOD \u0438 TIMEZONE\u00a0\u2014 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u044e\u0442, \u0437\u0430\u00a0\u043a\u0430\u043a\u043e\u0439 \u043f\u0440\u043e\u043c\u0435\u0436\u0443\u0442\u043e\u043a \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u043c\u044b \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u043d\u043e\u0432\u043e\u0441\u0442\u0438 \u0438 \u0432\u00a0\u043a\u0430\u043a\u043e\u043c \u0447\u0430\u0441\u043e\u0432\u043e\u043c \u043f\u043e\u044f\u0441\u0435 \u0438\u0445 \u0441\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u0435\u043c.\u041f\u0435\u0440\u0435\u0434 \u0437\u0430\u043f\u0443\u0441\u043a\u043e\u043c \u043d\u0430\u00a0Amvera (\u0438\u043b\u0438 \u0432\u00a0\u043b\u044e\u0431\u043e\u043c \u0434\u0440\u0443\u0433\u043e\u043c \u043c\u0435\u0441\u0442\u0435) \u043d\u0443\u0436\u043d\u043e \u043d\u0430\u0441\u0442\u0440\u043e\u0438\u0442\u044c \u0440\u0435\u0430\u043b\u044c\u043d\u044b\u0435 \u0437\u043d\u0430\u0447\u0435\u043d\u0438\u044f \u0447\u0435\u0440\u0435\u0437 \u0432\u0435\u0431-\u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441 \u0438\u043b\u0438 CLI Amvera.\u041b\u043e\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435\u0427\u0442\u043e\u0431\u044b \u043f\u043e\u043d\u0438\u043c\u0430\u0442\u044c, \u0447\u0442\u043e \u043f\u0440\u043e\u0438\u0441\u0445\u043e\u0434\u0438\u0442 \u0432\u043d\u0443\u0442\u0440\u0438 \u043d\u0430\u0448\u0435\u0433\u043e \u0441\u043a\u0440\u0438\u043f\u0442\u0430 \u043d\u0430 \u0443\u0434\u0430\u043b\u0451\u043d\u043d\u043e\u043c \u0441\u0435\u0440\u0432\u0435\u0440\u0435, \u043c\u044b \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0430\u0435\u043c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u044b\u0439 \u043c\u043e\u0434\u0443\u043b\u044c logging. \u042d\u0442\u043e \u043f\u043e\u0437\u0432\u043e\u043b\u0438\u0442 \u043d\u0435 \u00ab\u0442\u043e\u043d\u0443\u0442\u044c\u00bb \u0432 \u0441\u0442\u0435\u043d\u0435 \u0432\u044b\u0432\u043e\u0434\u0430 \u2013 \u0432\u0441\u0435 \u0432\u0430\u0436\u043d\u044b\u0435 \u0441\u043e\u0431\u044b\u0442\u0438\u044f \u0431\u0443\u0434\u0443\u0442 \u0441\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u044b \u0432 \u043b\u043e\u0433\u0438.import logginglogging.basicConfig(    level=logging.INFO,    format=&#8217;%(asctime)s &#8212; %(levelname)s &#8212; %(message)s&#8217;)logger = logging.getLogger(&#8216;bbc_news_scraper&#8217;)level=logging.INFO \u2014 \u0431\u0443\u0434\u0435\u043c \u043f\u043e\u043b\u0443\u0447\u0430\u0442\u044c \u0432\u0441\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f \u0443\u0440\u043e\u0432\u043d\u044f INFO \u0438 \u0432\u044b\u0448\u0435 (WARNING, ERROR).format=&#8217;%(asctime)s &#8212; %(levelname)s &#8212; %(message)s&#8217; \u2014 \u0432 \u043a\u0430\u0436\u0434\u043e\u0439 \u0441\u0442\u0440\u043e\u043a\u0435 \u043b\u043e\u0433\u0430 \u0431\u0443\u0434\u0435\u0442 \u0432\u0440\u0435\u043c\u044f, \u0443\u0440\u043e\u0432\u0435\u043d\u044c \u0438 \u0442\u0435\u043a\u0441\u0442 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u044f.logger = logging.getLogger(&#8216;bbc_news_scraper&#8217;) \u2014 \u0438\u043c\u0435\u043d\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u043b\u043e\u0433\u0433\u0435\u0440, \u0447\u0442\u043e\u0431\u044b \u043f\u0440\u0438 \u043c\u0430\u0441\u0448\u0442\u0430\u0431\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0438 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u043c\u043e\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0440\u0430\u0437\u0434\u0435\u043b\u044f\u0442\u044c \u043b\u043e\u0433\u0438 \u043f\u043e \u043c\u043e\u0434\u0443\u043b\u044f\u043c.\u0417\u0430\u0433\u0440\u0443\u0437\u043a\u0430 RSS-\u043b\u0438\u0441\u0442\u0430\u041e\u0441\u043d\u043e\u0432\u043d\u0430\u044f \u0437\u0430\u0434\u0430\u0447\u0430 \u043f\u0430\u0440\u0441\u0435\u0440\u0430\u00a0\u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c XML \u043f\u043e URL-\u0443 RSS. \u0417\u0434\u0435\u0441\u044c \u043d\u0430 \u043f\u043e\u043c\u043e\u0449\u044c \u043f\u0440\u0438\u0445\u043e\u0434\u0438\u0442 \u0431\u0438\u0431\u043b\u0438\u043e\u0442\u0435\u043a\u0430 requests:import requestsHEADERS = {&#8216;User-Agent&#8217;: &#8216;Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)&#8217;}def fetch_feed_xml(url):    &#171;&#187;&#187;    \u0417\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 RSS-\u0444\u0438\u0434 \u043f\u043e HTTP \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0441\u044b\u0440\u043e\u0435 \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u043c\u043e\u0435 XML.    \u0412 \u0441\u043b\u0443\u0447\u0430\u0435 \u043e\u0448\u0438\u0431\u043a\u0438 \u043b\u043e\u0433\u0438\u0440\u0443\u0435\u043c \u0435\u0435 \u0438 \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c None.    &#171;&#187;&#187;    try:        resp = requests.get(url, headers=HEADERS, timeout=10)        resp.raise_for_status()  # \u0431\u0440\u043e\u0441\u0438\u0442 \u0438\u0441\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435, \u0435\u0441\u043b\u0438 \u0441\u0442\u0430\u0442\u0443\u0441 \u2260200        return resp.content    except requests.RequestException as e:        logger.error(f&#187;\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f {url}: {e}&#187;)        return Nonetimeout=10 \u0437\u0430\u0449\u0438\u0449\u0430\u0435\u0442 \u043e\u0442 \u00ab\u0437\u0430\u0432\u0438\u0441\u0448\u0438\u0445\u00bb \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432.resp.raise_for_status() \u0441\u0440\u0430\u0437\u0443 \u00ab\u043f\u043e\u0439\u043c\u0430\u0435\u0442\u00bb HTTP-\u043e\u0448\u0438\u0431\u043a\u0438 (404, 500 \u0438 \u0442.\u200a\u0434.).\u041f\u0440\u0438 \u043b\u044e\u0431\u043e\u043c \u0438\u0441\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0438 \u043c\u044b \u043b\u043e\u0433\u0438\u0440\u0443\u0435\u043c \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0447\u0435\u0440\u0435\u0437 logger.error, \u0447\u0442\u043e\u0431\u044b \u0432 \u043b\u043e\u0433\u0430\u0445 \u0431\u044b\u043b\u043e \u0432\u0438\u0434\u043d\u043e, \u043a\u0430\u043a\u043e\u0439 \u0438\u043c\u0435\u043d\u043d\u043e \u0444\u0438\u0434 \u043d\u0435 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442.\u041f\u0430\u0440\u0441\u0438\u043d\u0433 RSS \u0438 \u0444\u0438\u043b\u044c\u0442\u0440\u0430\u0446\u0438\u044f \u043f\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443\u041f\u043e\u0441\u043b\u0435 \u0442\u043e\u0433\u043e \u043a\u0430\u043a \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0438 \u00ab\u0441\u044b\u0440\u043e\u0435\u00bb \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u043c\u043e\u0435 RSS, \u043d\u0443\u0436\u043d\u043e \u0438\u0437 \u043d\u0435\u0433\u043e \u0432\u044b\u0442\u0430\u0449\u0438\u0442\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0442\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u043d\u0430\u043c \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u044b\u00a0\u2014 \u0442\u043e \u0435\u0441\u0442\u044c \u0441\u0442\u0430\u0442\u044c\u0438 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0445 4\u00a0\u0447\u0430\u0441\u043e\u0432 \u0441\u043e \u0441\u043b\u043e\u0432\u043e\u043c \u00ab\u0422\u0440\u0430\u043c\u043f\u00bb \u0432\u00a0\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0435. \u0414\u043b\u044f\u00a0\u044d\u0442\u043e\u0433\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c xml.etree.ElementTree \u0438 \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u0443\u044e \u0443\u0442\u0438\u043b\u0438\u0442\u0443 \u0434\u043b\u044f \u0440\u0430\u0437\u0431\u043e\u0440\u0430 \u0434\u0430\u0442.import xml.etree.ElementTree as ETfrom email.utils import parsedate_to_datetimedef parse_feed_items(xml_content: bytes, cutoff: datetime, keyword: str) -&amp;gt; list[dict]:    &#171;&#187;&#187;    \u0420\u0430\u0437\u0431\u0438\u0440\u0430\u0435\u0442 XML, \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u0442  \u0438 \u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u0442 \u043f\u043e \u0434\u0430\u0442\u0435 \u0438 \u043a\u043b\u044e\u0447\u0435\u0432\u043e\u043c\u0443 \u0441\u043b\u043e\u0432\u0443.    \u0412\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u0442 \u0441\u043f\u0438\u0441\u043e\u043a \u0441\u043b\u043e\u0432\u0430\u0440\u0435\u0439 {&#8216;title&#8217;, &#8216;link&#8217;, &#8216;published&#8217;}.    &#171;&#187;&#187;    items = []    try:        root = ET.fromstring(xml_content)    except ET.ParseError as e:        logger.error(f&#187;\u041e\u0448\u0438\u0431\u043a\u0430 \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML: {e}&#187;)        return items    # \u0418\u0449\u0435\u043c \u0432\u0441\u0435 \u044d\u043b\u0435\u043c\u0435\u043d\u0442\u044b     for item in root.findall(&#8216;.\/\/item&#8217;):        title_el = item.find(&#8216;title&#8217;)        link_el = item.find(&#8216;link&#8217;)        pubdate_el = item.find(&#8216;pubDate&#8217;)        if not (title_el and link_el and pubdate_el):            continue        # \u041f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u0443\u0435\u043c pubDate \u2192 datetime \u0441 \u0447\u0430\u0441\u043e\u0432\u044b\u043c \u043f\u043e\u044f\u0441\u043e\u043c UTC        try:            pub_dt = parsedate_to_datetime(pubdate_el.text)            pub_dt = pub_dt.astimezone(TIMEZONE)        except Exception as e:            logger.warning(f&#187;\u041d\u0435 \u0443\u0434\u0430\u043b\u043e\u0441\u044c \u0440\u0430\u0441\u043f\u0430\u0440\u0441\u0438\u0442\u044c \u0434\u0430\u0442\u0443: {e}&#187;)            continue        # \u041e\u0442\u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u043c \u043f\u043e \u0432\u0440\u0435\u043c\u0435\u043d\u0438 \u0438 \u043f\u043e \u0441\u043b\u043e\u0432\u0443 \u201c\u0422\u0440\u0430\u043c\u043f\u201d        if pub_dt &amp;gt;= cutoff and keyword.lower() in title_el.text.lower():            items.append({                &#8216;title&#8217;: title_el.text.strip(),                &#8216;link&#8217;: link_el.text.strip(),                &#8216;published&#8217;: pub_dt.strftime(&#8216;%Y-%m-%d %H:%M:%S %Z&#8217;)            })    return items\u041c\u044b \u0438\u0449\u0435\u043c \u0432\u0441\u0435 &#171; \u0432\u043d\u0443\u0442\u0440\u0438 RSS \u0438 \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0443 \u043a\u0430\u0436\u0434\u043e\u0433\u043e \u0435\u0441\u0442\u044c \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043e\u043a, \u0441\u0441\u044b\u043b\u043a\u0430 \u0438 \u0434\u0430\u0442\u0430.\u0421 \u043f\u043e\u043c\u043e\u0449\u044c\u044e parsedate_to_datetime \u0438\u0437 email.utils \u043a\u043e\u043d\u0432\u0435\u0440\u0442\u0438\u0440\u0443\u0435\u043c \u0441\u0442\u0440\u043e\u043a\u0443 \u0432\u0438\u0434\u0430 Fri, 07 Aug 2025 12:34:56 GMT \u0432 datetime \u0441 \u043f\u043e\u044f\u0441\u043e\u043c UTC.\u0421\u0440\u0430\u0432\u043d\u0438\u0432\u0430\u0435\u043c \u0434\u0430\u0442\u0443 \u043f\u0443\u0431\u043b\u0438\u043a\u0430\u0446\u0438\u0438 \u0441 cutoff (\u0442\u0435\u043a\u0443\u0449\u0438\u043c \u0432\u0440\u0435\u043c\u0435\u043d\u0435\u043c \u043c\u0438\u043d\u0443\u0441 4 \u0447).\u0414\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u043e\u0435\u0443\u0441\u043b\u043e\u0432\u0438\u0435\u00a0\u2014 \u0432\u00a0\u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0435 \u0434\u043e\u043b\u0436\u043d\u043e \u0432\u0441\u0442\u0440\u0435\u0447\u0430\u0442\u044c\u0441\u044f \u0441\u043b\u043e\u0432\u043e \u00ab\u0422\u0440\u0430\u043c\u043f\u00bb (\u0431\u0435\u0437 \u0443\u0447\u0435\u0442\u0430 \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430).\u0422\u0430\u043a \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0435 \u043d\u043e\u0432\u043e\u0441\u0442\u0438, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0441\u0432\u0435\u0436\u0438\u0435 \u0438 \u0440\u0435\u043b\u0435\u0432\u0430\u043d\u0442\u043d\u044b\u0435 \u043d\u0430\u0448\u0435\u0439 \u0442\u0435\u043c\u0435.\u041e\u0442\u043f\u0440\u0430\u0432\u043a\u0430 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439 \u0432 Telegram\u041a\u043e\u0433\u0434\u0430 \u0441\u043f\u0438\u0441\u043e\u043a \u0441\u0432\u0435\u0436\u0438\u0445 \u0441\u0442\u0430\u0442\u0435\u0439 \u0433\u043e\u0442\u043e\u0432, \u0444\u043e\u0440\u043c\u0438\u0440\u0443\u0435\u043c \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u043c \u0435\u0433\u043e \u0432 \u0432\u0430\u0448 \u043a\u0430\u043d\u0430\u043b \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e Telethon:from telethon import TelegramClientfrom telethon.sessions import StringSessionasync def send_to_telegram_channel(message: str):    &#171;&#187;&#187;    \u0410\u0441\u0438\u043d\u0445\u0440\u043e\u043d\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u044f\u0435\u0442 \u0442\u0435\u043a\u0441\u0442 \u0432 Telegram-\u043a\u0430\u043d\u0430\u043b.    \u0412\u0441\u0435 \u043b\u0438\u0447\u043d\u044b\u0435 \u0434\u0430\u043d\u043d\u044b\u0435 (API_ID, API_HASH, BOT_TOKEN, SESSION) \u0432 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u0437\u0430\u043c\u0435\u043d\u0435\u043d\u044b.    &#171;&#187;&#187;    try:        bot_client = TelegramClient(            StringSession(BOT_SESSION_STRING) if BOT_SESSION_STRING else &#8216;session_bot&#8217;,            API_ID, API_HASH        )        await bot_client.start(bot_token=BOT_TOKEN)        entity = await bot_client.get_entity(CHANNEL_LINK)        # link_preview=False \u2013 \u0447\u0442\u043e\u0431\u044b \u043d\u0435 \u043f\u043e\u0434\u0442\u044f\u0433\u0438\u0432\u0430\u043b\u0438\u0441\u044c \u0431\u043e\u043b\u044c\u0448\u0438\u0435 \u043a\u0430\u0440\u0442\u0438\u043d\u043a\u0438        await bot_client.send_message(entity, message, link_preview=False)        logger.info(&#171;\u0421\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0443\u0441\u043f\u0435\u0448\u043d\u043e \u043e\u0442\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u043e \u0432 Telegram&#187;)    except Exception as e:        logger.error(f&#187;\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u0440\u0438 \u043e\u0442\u043f\u0440\u0430\u0432\u043a\u0435 \u0432 Telegram: {e}&#187;)    finally:        if bot_client.is_connected():            await bot_client.disconnect()StringSession \u0438\u043b\u0438 \u043e\u0431\u044b\u0447\u043d\u0430\u044f \u0441\u0435\u0441\u0441\u0438\u044f\u00a0\u2014 \u0445\u0440\u0430\u043d\u0438\u0442 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \u043e \u0432\u0430\u0448\u0435\u043c \u0431\u043e\u0442\u0435.start(bot_token=&#8230;) \u0434\u0435\u043b\u0430\u0435\u0442 \u0430\u0432\u0442\u043e\u0440\u0438\u0437\u0430\u0446\u0438\u044e \u043f\u043e \u0442\u043e\u043a\u0435\u043d\u0443.get_entity \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043d\u0430\u0445\u043e\u0434\u0438\u0442 \u043d\u0443\u0436\u043d\u044b\u0439 \u043a\u0430\u043d\u0430\u043b \u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0435.\u041f\u043e\u043b\u043d\u044b\u0439 \u043a\u043e\u0434 (\u0441\u043a\u0435\u043b\u0435\u0442 \u0441\u043a\u0440\u0438\u043f\u0442\u0430)import osimport requestsimport xml.etree.ElementTree as ETfrom datetime import datetime, timedeltaimport pytzfrom email.utils import parsedate_to_datetimeimport loggingimport asynciofrom telethon import TelegramClientfrom telethon.sessions import StringSessionimport time# ========== CONFIGURATION ==========DATA_FOLDER = &#8216;\/data&#8217;RSS_FEEDS = [    &#8216;https:\/\/feeds.bbci.co.uk\/news\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/world\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/business\/rss.xml&#8217;,    &#8216;https:\/\/feeds.bbci.co.uk\/news\/technology\/rss.xml&#8217;,]# Telegram configurationAPI_ID = int(os.getenv(&#8216;API_ID&#8217;, 2122521))API_HASH = os.getenv(&#8216;API_HASH&#8217;, &#8216;d27621d9925562342baefc013e&#8217;)BOT_TOKEN = os.getenv(&#8216;BOT_TOKEN&#8217;, &#8216;759406329:32522CPhmU&#8217;)CHANNEL_LINK = os.getenv(&#8216;CHANNEL_LINK&#8217;, &#8216;https:\/\/t.me\/+vR323w55y&#8217;)BOT_SESSION_STRING = os.getenv(&#8216;BOT_SESSION_STRING&#8217;)HEADERS = {&#8216;User-Agent&#8217;: &#8216;Mozilla\/5.0 (compatible; BBCNewsScraper\/1.0)&#8217;}TIMEZONE = pytz.utcPERIOD = timedelta(hours=4) # \u041f\u0435\u0440\u0438\u043e\u0434 &#8212; 4 \u0447\u0430\u0441\u0430MOSCOW_TZ = pytz.timezone(&#8216;Europe\/Moscow&#8217;)# ========== LOGGING ==========logging.basicConfig(    level=logging.INFO,    format=&#8217;%(asctime)s &#8212; %(levelname)s &#8212; %(message)s&#8217;)logger = logging.getLogger(&#8216;bbc_news_scraper&#8217;)# ========== UTILS ==========def ensure_dir(path):    os.makedirs(path, exist_ok=True)# ========== FETCH FEED ==========def fetch_feed_xml(url):    try:        resp = requests.get(url, headers=HEADERS, timeout=10)        resp.raise_for_status()        return resp.content    except requests.RequestException as e:        logger.error(f&#187;\u041e\u0448\u0438\u0431\u043a\u0430 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u044f {url}: {e}&#187;)        return None# ========== PARSE FEED ITEMS ==========def parse_feed_items(xml_content, cutoff):    items = []    try:        root = ET.fromstring(xml_content)    except ET.ParseError as e:        logger.error(f&#187;\u041e\u0448\u0438\u0431\u043a\u0430 \u0440\u0430\u0437\u0431\u043e\u0440\u0430 XML: {e}&#187;)        return items    for item in root.findall(&#8216;.\/\/item&#8217;):        title_el =&#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-475188","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/475188","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=475188"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/475188\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=475188"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=475188"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=475188"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}