LINUX.ORG.RU

Сообщения Xintrea

 

Где прописывается запуск Qt Creator?

Форум — Development

Сегодня менял HDD, переехал на новый, все скопировал через dd, но произошло странное...

В системе был вручную установлен Qt 5.12.6 из официального инсталлятора с qt.io. Все было установлено в каталог /opt/qt_5_12_6. Там же и эта версия и лежит, никуда ничего не перемещалось. В этом наборе имеется Qt Creator 4.10.2.

Так вот, еще сегодня утром, если в меню DE тыкнуть Старт - Разработка - Qt Creator, то запускался Qt Creator 4.10.2.

А после замены HDD, если тыкнуть в меню тот же пункт, то откроется Qt Creator 4.6.2. Это старый Creator, который был давно еще установлен из официального репозитария дистрибутива вместе с Qt 5.11.0. Эти версии тоже как были когда-то установлены, так и стоят, никто ничего на диске не трогал.

В пункте меню используется команда:

qtcreator %F

Вопрос: почему изменился запускаемый Qt Creator? Как вернуть запуск Qt Creator 4.10.2 из /opt/qt_5_12_6 ?

 , ,

Xintrea
()

Миграции в условиях паранойи - полная проверка структуры таблиц

Форум — Development

Привет, народ.

Требуется тут сделать нестандарную самописную подсистему миграции в БД PostgreSQL 9.6 с учетом параноидальности начальства.

В стандартном виде миграция осуществляется по алгоритму: где-то хранится номер последнего обновления, и если есть новое обновление, то оно применяется поверх тех данных и структуры что есть в БД. При этом все система полагается на то, что если известна версия текущей структуры, то однозначно известна и сама структура, и эта структура правильная. И поэтому миграция применяется без реальной проверки структуры таблиц. И во всех разумных сценариях это работает.

Теперь рассмотрим параноидальный сценарий:

Предполагается, что в любой момент может появиться «тайный криворукий вредитель», который что-то сделает с таблицами в БД. С самими БД он ничего не сделает (не спрашивайте почему, никто этого не знает), а вот с таблицами - сделает. Поменяет тип столбца, переименует столбец или вовсе удалит, уберет индекс или еще что-то эдакое. Или вообще удалит таблицу.

Поэтому, в миграции надо автоматически проверять полную структуру таблиц, которая сформировалась в результате всех предыдущих миграций. И если структура неправильная, то удалять все данные, и прогонять все миграции заново с самой первой (да, вот такой странный подход).

Вопрос 1: как автоматически «вычислять» структуру таблиц, сформировавшуюся за все миграции, чтобы иметь возможность ее проверить?

Вопрос 2: как непосредственно проверить структуру таблиц на соответствие «вычисленной» на предыдущем шаге структуре?

Есть ли готовые SQL-команды или подходы для таких действий?

Используемые инструменты: C++, Qt. Никаких этих ваших яв, питонов, пхп и сторонних библиотек быть не должно.

PS: Я не прикалываюсь, я все это спрашиваю абсолютно серьезно.

 , , , ,

Xintrea
()

Свежая задачка с Yandex Backend Tour 2023 - «Опытная команда»

Форум — Talks

Эта тема создается для тех, кто завтра еще будет иметь возможность порешать задачки с Yandex Backend Tour 2023.

Условие задачи «Опытная команда» я не разглашаю (покажу ее после 23:59 19.11.2023), но хочу написать что имели в виду авторы. У меня плохо с пониманием других людей, а когда решение нужно делать на время, когнитивные функции напрочь отключаются.

Так вот, когда вы будете пытаться понять, почему в разборе примера авторы написали вот это:

----- 8< -----

В момент времени T=2 к команде присоединяется Victor. Опытность Ivan и Anton теперь равна 1, опытность Victor равна 0. Разность между суммарной опытностью оставшихся членов команды и опытностью самого опытного члена команды равна (1+0)−1=0. Добавляем к ответу Anton 0

----- 8< -----


То возможно вас наведет на правильную мысль следующее объяснение:


----- 8< -----

Ребята из Яндекса не написали в условии задачи, как определяется самый опытный. Зато написали в формате вывода: «Если членов команды с максимальной опытностью несколько, нужно вывести того, чье имя лексикографически минимально».

За этой фразой, видимо, скрывается мысль, что этот принцип используется не только при выводе. Видимо, подразумевалось, что данный принцип надо использовать и в других местах программы. (Для меня это не очевидно).

Поэтому я, наивный, и не понял, что когда считается опытность самого опытного члена, то если получается более одного члена с одинаково высоким опытом, то надо выбрать лексикографически самого опытного, и суммарный опыт посчитать без этого человека.

Другими словами, нужно учитывать что «лексикографический выбор» надо делать не только при выборе, какого кандидата показать в результирующей строке. Но и это же нужно делать и в основном коде, когда считается опытность команды. И значение этого человека не включать в суммарный опыт команды.

----- 8< -----


Вот, возможно кому-то поможет. Я потратил на попытки понять условие много времени, потому что было обидно что код написан, все разложено по полочкам, а я не догоняю что от меня конкретно хотят.

PS: Задача даже не на алгоритмы, а просто на корректное написание простого кода, который парсит входные данные по некоторым правилам.

PPS: И да, я не прошел.

 ,

Xintrea
()

Как настроить в VSCode нестандартный запуск Python-скрипта? А чтобы дебаг работал?

Форум — Development

Есть специальная команда для запуска python-скрипта:

cat file.txt | python3 sample.py

1. Как сделать так, чтобы она запускалась по Ctrl+F5 вместо стандартного запуска скрипта?

2. А как сделать, чтобы и дебаг запускался с такой же командой?

VSCode v. 1.81.1, плагин - стандартный Python IntelliSence (Pylance) от Microsoft v.2023.14.0.

 , , , ,

Xintrea
()

Как в Python написать цикл с постусловием?

Форум — Development

Классический цикл for(инит; действие при каждой итерации; условие продолжения) в C/C++ - это, по-сути цикл с пост-условием.

А как написать for-цикл с пост-условием в Python? Если пользоваться range(), то range() «запекает» перебираемые значения при старте цикла.

А нужно, чтобы цикл бежал не до фиксированного значения, а до того, пока будет выполняться условие. Как это сделать?

Только через while-цикл и никак иначе?

Я пробовал через for и генератор с условием, наложенным на range, но не получается:

for i in ( x for x in range(0, len(s)) if x<(len(s)-maxLen)) ):
Так тоже в процессе работы цикла количество итераций не уменьшается.

 , ,

Xintrea
()

Как в Google/Yandex найти страницу, на которой упоминается заданная ссылка?

Форум — General

Google все время пытается проиндексировать страницы, которых нет на сайте webhamster.ru ужо несколько лет как.

Вот примеры таких ссылок:

http://webhamster.ru/mytetrashare/index/mtb131/1490546683q10675zpp6
https://webhamster.ru/mytetrashare/index/mtb176/14982150565oqhyblndu
https://webhamster.ru/mytetrashare/index/mtb189/1490267599a1aqopr8ha

То есть, где-то, или на самом webhamster.ru или на каких-то других сайтах эти ссылки существуют.

Вопрос: какой запрос надо дать в Google или в Yandex чтобы найти в Интернете страницы, которые содержат вышеперечисленные линки?

 , , ,

Xintrea
()

Миграции: как в самой таблице хранить номер версии структуры таблицы?

Форум — Development

Для создания наколеночной системы миграции таблиц в PostgreSQL нужно мне где-то хранить номер версии структуры таблицы.

Существует ли способ где-то сохранить номер версии структуры таблицы в самой таблице?

То есть, не создавать в БД специальную таблицу, в которой будут храниться имена таблиц и их текущие версии структуры, а хранить номер версии в какой-то сопровождающей таблицу информации?

 , , , ,

Xintrea
()

Что такое PulseAudio? Как оно работает?

Форум — General

Имеется следующая цитата:

PulseAudio - звуковой сервер в виде демона, запускаемый, что интересно, от имени обычного пользователя. Создает слой абстракции звуковых потоков для возможности различной коммутации, принимает по собственному API звуковой поток от приложений и воспроизводит его через ALSA. Так же PulseAudio создает для ALSA специальный порт, через которые может получать данные, отправляемые приложениями в сам ALSA, и такое поведение настраивается по-дефолту при установке PulseAudio. Считается надстройкой над ALSA, так как без ALSA его использование бессмысленно (будет работать только комутация, а звука не будет).


Имеются ли в ней грубые фактические ошибки, вводящие читателя в заблуждение? Сам пользуюсь PulseAudio, и конфигурю его по необходимости «чтоб работало», но понимание внутреннего устройства ограничено общими фразами из околозвуковых статей.

 , , , ,

Xintrea
()

Все уже посмотрели новость как НИИЭТ перемаркировывает китайские MOSFET на российские GaN?

Форум — Talks

Максим Горшенин снова бомбанул. После разборок с чипом Миландра, который заменяет три кусочка проволоки в мониторах LightCom, ему подсказали подумать вот о чем... Как так получается, что российская автомобильная зарядка с разработанными в России GaN-транзисторами от Воронежского института электронной техники имеет абсолютно ту же плату, что и некоторые китайские зарядки с классическими MOSFET-тами? Почему совершенно не меняется схемотехника? Не кажется ли это странным?

По счастливой случайности у Горшенина на руках оказалось две зарядки, которые ему лично передали представители НИИЭТ, на которых имеются транзисторы с логотипом НИИЭТ. Об этих зарядках Горшенин даже снимал ролик. Каково же было его удивление, когда внимательнее посмотрев под лупой на эти транзисторы он обнаружил, что, верхняя поверхность корпуса транзистора полностью снята лазером, и поверх, этим же лазером, сделана маркировка НИИЭТ!

Не поверив собственным глазам, Горшенин позвонил в НИИЭТ и поинтересовался, а как так получилось, что на упаковке написано что это зарядка с GaN-транзисторами, а оказалось, что это простая зарядка с китайскими MOSFET-транзисторами? Представители НИИЭТ в телефонном разговоре сообщили, что это все китайцы виноваты, и вложили в коробочки не те зарядные устройства. На вопрос почему транзисторы пиленные, в НИИЭТ ответить не смогли. Но очень хотели, чтобы Максим не публиковал эту информацию, подумал, поговорил с гендиректором института и не делал поспешных выводов.


Веселая новость с фотографиями:
https://webhamster.ru/site/page/index/main/news/720

Видео 1 - Прекрасные зарядки с GaN транзисторами:
https://www.youtube.com/watch?v=ilNeKlgBpwU

Видео 2 - Походу, это развод покупателей:
https://www.youtube.com/watch?v=E53Mo_rHDqs

 , , перелицовка, ,

Xintrea
()

Как закрыть тред при выходе из Tkinter-приложения?

Форум — Development

Имеется Tkinter-приложение, в котором есть:

  • Основное окно
  • Дополнительное окно, которое запускается в отдельном треде. Оно содержит кнопку Panic.

Задача проста: при завершении работы программы любым способом (например, по сочетанию Alt+F4), нужно корректно завершить программу.

Я сделал минимальный пример. При завершении его работы появляется ошибка:

Запускается тред для окна с кнопкой остановки
Внутри треда с окном-кнопкой Panic
Устанавливается флаг на остановку треда с кнопкой Panic
Завершен тред _threadCodeSmallWindow
Тред с кнопкой Panic полностью завершен

Tcl_AsyncDelete: async handler deleted by the wrong thread
Аварийный останов

То есть, нужно, чтобы этой ошибки не возникало.

Если в примере закомментировать последнюю команду main.destroySmallWindow(), то ошибки не будет. Но и программа не будет завершаться по Alt+F4, потому что не завершен тред с дополнительным окном.

Я пробовал перед вызовом rootWindow.mainloop() добавить команду:
rootWindow.protocol("WM_DELETE_WINDOW", onDeleteWindow)

И функцию при закрытии писал такую:
def onDeleteWindow():
    main.destroySmallWindow() # Завершение треда
    rootWindow.destroy() # Закрытие основного окна

Но и тогда все равно ошибка сохраняется.

Вопрос: как корректно завершить программу?

Код примера:
#!/usr/bin/python3

import tkinter as tk
import time
from threading import Thread, Event

class MainFrame(tk.Frame):

    def __init__(self, parent):

        self.stopThreadFlag = Event()
        self.smallWindow = None
        self.threadSmallWindow = None

        # Инициализация базового класса рамки
        super(MainFrame, self).__init__(parent)

        # Основная рамка
        frame = tk.Frame(self, relief=tk.RAISED, borderwidth=1)
        frame.pack(fill=tk.BOTH, expand=True)

        # Надпись
        label = tk.Label(frame, text="Содержимое окна")
        label.pack(anchor=tk.W)

        # Создается окошко-кнопка с кнопкой Panic в отдельном потоке
        self._createSmallWindow()


    # Создание окошка с кнопкой Panic в отдельном потоке
    def _createSmallWindow(self):
        print("Запускается тред для окна с кнопкой остановки")
        self.stopThreadFlag.clear() # Очищается флаг прекращения работы потока
        self.threadSmallWindow = Thread(target = self._threadCodeSmallWindow)
        self.threadSmallWindow.start()


    # Код, выполняемый внутри треда
    def _threadCodeSmallWindow(self):
        print("Внутри треда с окном-кнопкой Panic")

        # Создается окошко с кнопкой Panic
        self.smallWindow= tk.Tk() # = tk.Toplevel(Parameter().rootWindow)
        self.smallWindow.resizable(0, 0) # Запрещает изменение размера
        self.smallWindow.overrideredirect(1) # Отключается все оформление окона

        smallWindowFrame = tk.Frame( self.smallWindow )
        smallWindowFrame.pack(fill="both", expand=True)

        button=tk.Button(smallWindowFrame, text="Panic", borderwidth=0, relief=tk.FLAT)
        button.pack(fill=tk.BOTH, anchor="center", expand=True)

        self.smallWindow.update() # Прорисовка окна с кнопкой Panic

        # Вместо mainloop() используется цикл, который завершается при установке флага
        while not self.stopThreadFlag.is_set():
            if self.smallWindow != None:

                # ... Всякие действия ...

                time.sleep(0.5) # Разгрузка работы, чтобы не выжирался весь процессор

        if self.smallWindow != None:
            self.smallWindow.destroy()

        print("Завершен тред _threadCodeSmallWindow")


    # Уничтожение окна и треда с кнопкой Panic
    def destroySmallWindow(self):
        if self.smallWindow != None:
            print("Устанавливается флаг на остановку треда с кнопкой Panic")
            self.stopThreadFlag.set()

            self.threadSmallWindow.join() # Ожидание завершения треда с кнопкой Panic
            print("Тред с кнопкой Panic полностью завершен")


if __name__ == "__main__":

    # Запуск Tk-подсистемы графических виджетов и создание основного окна
    rootWindow = tk.Tk()
    rootWindow.geometry("320x200+0+0")

    # Создание основной рамки с интерфейсом приложения
    main = MainFrame( rootWindow )
    main.pack(fill="both", expand=True)

    # Основной цикл GUI-интерфейса
    rootWindow.mainloop()

    # Вызов завершения треда с окошком Panic
    # Без этого вызова программа не завершится
    main.destroySmallWindow()

 , , ,

Xintrea
()

У кого-нибудь открывается в LibreOffice CSV файл ОКАТО Росстата?

Форум — Desktop

Файл с классификтором ОКАТО:

https://rosstat.gov.ru/opendata/7708234640-7708234640-okato/data-20230803-str...

Не открывается в LibreOffice 7.0.4.2 - программа зависает.

В MS Office все в порядке.

У кого-то LO открывает этот файл?


Перемещено hobbit из general

 

Xintrea
()

Тормозит открытие окна файлового диалога в KDE/LXDE? Я знаю решение.

Форум — General

Всего год мучений, и я разобраться как побороть застаревший баг Qt-шного окна открытия файла.

Баг и решение написаны здесь: Исправление ошибки в KDE/LXDE если долго открывается файловый диалог

Пользуйтесь, если у кого похожая проблема.

 , , , диалог,

Xintrea
()

Эксперимент со звуком в Audacity: синус плюс синус = пшпшпш

Форум — Talks

Открываю Audacity, создаю две моно-дорожки с синусоидальным сигналом на частотах 261 Гц (До) и 440 Гц (Ля).

Создать - Тон
Волноформа: Синусоидальная
Частота: 261 (или 440)
Амплитуда: 0,5
Длительность: 1 сек
По отдельности дорожки звучат красивым мягким синусом.

Но вместе получается жуткий пердеж. Ну не должно такого быть. Ну как так то? Что мешает двум синусам звучать вместе красиво? Вроде как это большая септима, и по классификации должна быть несовершенным, но все-таки консонансом. А тут даже не консонанс и не диссонанс, а тупо пердеж.

На ЦАП звуковой карточки и частоту дискретизации тоже вроде не погрешишь, она в 100 раз выше, а по теореме Котельникова нам достаточно было бы меньше 1000 Гц чтоб услышать все что нам нужно.

Почему такой эффект?

 волна, , ,

Xintrea
()

Как после chroot вызвать бинарники, находящиеся в основной системе?

Форум — General

Привет, народ.

Загрузился с System Rescue Cd. И нужно мне выполнить команду efibootmgr, и она есть на загруженном CD-Rom диске. Но чтобы она выполнилась, надо через chroot переключиться на каталог /mnt/sda3, куда смонтирована система, установленная на жестком диске.

Проблема в том, что в этой операционке нет efibootmgr. И после переключения через chroot, утилита efibootmgr становится недоступной.

Вопрос: есть ли возможность после chroot вызывать утилиты, находящиеся в основной загруженной системе?

 ,

Xintrea
()

Пятница. Давайте заставим работать osci-render и james.ball.sh/oscilloscope в Linux

Форум — General

Есть такой проект:

https://github.com/jameshball/osci-render

Это генератор звуковых стерео-сигналов, которые будучи поданные на осциллограф, рисуют разные картинки. В проект входят, как минимум, две программы:

1. osci-render - программа на Java, имеющая крутилки-вертелки и подгрузку векторных картинок. В реальном времени должна гудеть в динамик в зависимости от картинки для осцилографа, которую она расчитывает. При старте поднимает локальный порт 42988. Предполагается, что к нему может подключиться программный осциллограф чтобы визуализировать картинку.

2. https://james.ball.sh/oscilloscope - программный осциллограф, работающий в браузере. Для отображания картинки использует Canvas. При старте подключается к веб-сокету ws://127.0.0.1:42988, созданному osci-render. Должен показывать картинку, генерирующуюся osci-render-ом.

Проблема в том, что хоть проект и кроссплатформенный, разработчик практически не проверяет работу в Linux. Поэтому ничего не работает.

Во-первых, криво стартует сама программа osci-render. Чтобы увидеть рабочий интерфейс osci-render, надо поизвращаться с jackd, иначе интерфейс зависает. Но это кое-как решаемая проблема.

Проблема номер два: сама связка osci-render и james.ball.sh/oscilloscope не работает. И вот это уже проблема, которую я не могу решить. Я посмотрел netstat - порт 42988 создается. Я посмотрел wireshark - обмен между программным осциллографом и osci-render идет. Но этот осциллограф ничего не показывает, только точку в центре экрана.

Вот эту проблему и надо решить.

Подробности настройки здесь: https://webhamster.ru/mytetrashare/index/mtb0/16907522569aygrnabey

 , , , ,

Xintrea
()

Летний фестиваль Undefined 2023

Новости — Конференции и встречи
Летний фестиваль Undefined 2023
Группа Конференции и встречи

Undefined 2023 - это теплый, ламповый фестиваль DIY-электроники, демосцены, ретро-компьютеров и всего остального. Летняя сессия, традиционно, проходит в пригороде Санкт-Петербурга в поселке Рощино. Фестиваль проводится 26-27 августа 2023 г., а это значит, что остался ровно месяц до старта мероприятия.

( читать дальше... )

>>> Подробности

 , , ,

Xintrea
()

А ну-кось, лорчане, соколики мои! Навалитесь же ЛОР-эффектом на мой сайт чтоб проверить его устойчивость!

Форум — Talks

Привет, народ

Восстановил наконец я работу своего супер-сайта webhamster.ru, каковой существует ужо без малого 16 лет.

Полтора месяца назад сайт был удален на хостинге без возможности восстановления виртуалки. После этого факапа у меня все небыло времени восстановиться, ну и вот наконец добрался я до консоли и раскатал сайт заново. Для такого дела я даже написал Ansible-плейбуки, на всякий случай. Теперь если придется заново разворачивать из бекапа, все действия будут происходить автоматически, и не нужно будет вспоминать что где надо докрутить чтобы нормально настроить (всего, кстати, чтобы раскатать сайт, нужно выполнить около ~150 действий, и ничего не забыть).

В общем, у меня просьба: перейти на сайт https://webhamster.ru и прощелкать пару-тройку страниц. Нет, скрипты писать не надо (да кто ж вас остановит), пусть нагрузка будет естественная.

Посмотрим, выдержит ли этот инстанс поток пользователей с ЛОР-а.

 , лор-эффект, ,

Xintrea
()

Где прописать остановку сервиса snapd при старте системы?

Форум — Admin

Имеется сервер Debian Linux 11.

Для работы certbot от Letsencrypt на сервере установлен snapd, так как это единственный официальный способ установки бота сертификатов. Но этот snapd нужен только раз в месяц, когда срабатывает скрипт проверки и обновления SSL сертификатов. В остальное время snapd только жрет память и ресурсы процессора.

В скрипте запуска certbot прописаны команды включения сервисов snapd и snapd.socket перед запуском утилиты, и команды выключения этих сервисов после того как утилита certbot закончит работать.

То есть, snapd в обычном состоянии всегда должен быть потушен.

Чтобы snapd не стартовал при запуске, я дал команды:

systemctl disable snapd
systemctl disable snapd.socket
И перегрузился.

Но оказалось, что после перезагрузки сервис snapd все равно стартует и висит в памяти.

Я не понимаю, зачем тогда нужен systemctl, если он все равно ничем не управляет? Как оставить snapd в системе, но не давать ему включаться при старте? Что где еще надо докручивать?



Перемещено hobbit из general

 , , , ,

Xintrea
()

Как настроить отправку писем по SMTP через mail.google.com в 2023 году?

Форум — General

Сейчас восстанавливаю работу своего сайта, и занялся вопросом отправки писем.

Год назад все работало согласно данной инструкции:
https://webhamster.ru/site/page/index/articles/comp/149

Но теперь ssmtp показывает ошибку:

sendmail: Authorization failed (535 5.7.8  https://support.google.com/mail/?p=BadCredentials v6-20020a2e9246000000b001b6e... - gsmtp)

Я начал разбираться, и узнаю, что:

Чтобы защитить ваш аккаунт, начиная с 30 мая 2022 года Google больше не поддерживает сторонние приложения и устройства, которые предлагают войти в аккаунт Google только с помощью имени пользователя и пароля.

https://support.google.com/accounts/answer/6010255?hl=ru


В то же время, официальная документация Google говорит, что можно включить работу с «ненадежными приложениями». И я даже нашел прямой линк на эту настройку: https://myaccount.google.com/lesssecureapps

Однако на этой странице написано следующее:

Ненадежные приложения, у которых есть доступ к аккаунту
Эта настройка недоступна в вашем аккаунте.


Я читаю почему это мне настройка недоступна, и написано, что эта настройка доступна только для тех аккаунтов, на которых не включена двухфакторная аутентификация. Но дело в том, что она у меня и не включена:

Google аккаунт - Безопасность - Вход в Google - Двухэтапная аутентификация - Двухэтапная аутентификация отключена.


Все, на этих взаимоисключающих параграфах я запутался окончательно. Что еще нужно Гуглу? В каких местах что еще надо крутить чтобы отправка почты по SMTP заработала?

 , , ,

Xintrea
()

Добавление секции non-free в APT через Ansible - почему удаляются строки?

Форум — Development

Имеется исходный файл /etc/apt/sources.list :

deb http://ftp.ru.debian.org/debian bullseye main contrib
deb-src http://ftp.ru.debian.org/debian bullseye main contrib

deb http://security.debian.org/debian-security bullseye-security main contrib
deb-src http://security.debian.org/debian-security bullseye-security main contrib
И надо в него добавить секцию non-free.

Придумал такой код:
   - name: "Добавление секции non-free в APT"                                                                              
     replace:                                                                                                              
       dest: /etc/apt/sources.list                                                                                         
       regexp: '(deb|deb-src)\s*(http://.*debian.*)\s*(.?)\s*main contrib\s.*(?!non-free)'                                 
       replace: '\1 \2 \3 main contrib non-free'
В результате в файле, за каким-то хреном, исчезают строки deb-src:
deb http://ftp.ru.debian.org/debian bullseye   main contrib non-free

deb http://security.debian.org/debian-security bullseye-security   main contrib non-free

Почему так то? Какое правильное регулярное выражение должно быть?

 , , ,

Xintrea
()

RSS подписка на новые темы