Расширенная версия: Я создал Телеграм бота (FYTT), который ищет Телеграм каналы всех ваших подписок на ютубе / Хабр (habr.com)
Бот - https://t.me/FYTTproject_bot
Идея пришла, конечно же, после замедления ютуба. Многие блогеры стали активно кричать, мол переходите в тг, не потеряйте и я решил сделать удобный инструмент для быстро поиска тг-каналов своих любимых авторов.
Опыта мало в разработке, поэтому стек особо не выбирал:
Node.js, библиотека telegraf.js
MongoDB, mongoose
Express.js
Google API + Lemnos Api
Алгоритм работы бота, такой:
Пользователь пишет /start, ему выдается приветственное с 2 кнопками: "Найти Youtube-каналы в Telegram" и "Связаться Youtube-канал с Telegram-каналом". Также, пользователь сразу же добавляется в базу данных с сохранение его chatId, для дальнейшей связи, в случае чего и awatingChannels, о котором рассказал ниже.
bot.start(async (ctx) => {
const chatId = ctx.chat.id;
let chat = await Analytics.findOne({ chatId: chatId })
if (chat === null) {
try {
let username = ctx.message.chat.username
let newChat = new Analytics({
chatId: ctx.message.chat.id,
awatingChannels: true,
status: "member",
count: 0
})
await newChat.save()
} catch {
let newChat = new Analytics({
chatId: ctx.message.chat.id,
awatingChannels: true,
status: "member",
count: 0
})
await newChat.save()
}
} else {
chat.awatingChannels = true
await chat.save()
}
await setBotCommands()
ctx.replyWithHTML('<b>Приветствуем вас в нашем сервисе поиска Telegram-каналов ютуберов!</b>\nБот безопасен, так как представляет собой открытый исходный код, который может посмотреть каждый желающий. (/faq или пишите @vitosperansky)\n\nПоддержать проект: https://www.donationalerts.com/r/vitosperansky\n\nВыберите опцию:', Markup.inlineKeyboard([
[Markup.button.callback('Найти YouTube-каналы в Telegram', 'find_channels')],
[Markup.button.callback('Связать YouTube-канал с Telegram-каналом', 'link_channel')]
]), {
disable_web_page_preview: true
});
});
При нажатие на первую, пользователь получает кнопку с ссылкой на авторизацию в Google, после авторизации отправляется ответ на сервер на "/oauth2callback", запуская функцию поиска:
async function listSubscriptions(auth) {
const service = google.youtube('v3');
let subscriptions = [];
let nextPageToken = null;
try {
do {
const response = await service.subscriptions.list({
auth: auth,
part: 'snippet',
mine: true,
maxResults: 50, // 50 - лимит Google API
pageToken: nextPageToken // Устанавливаем токен страницы, если есть
});
// Добавляем текущую партию подписок
subscriptions = subscriptions.concat(response.data.items.map(item => ({
title: item.snippet.title,
channelId: item.snippet.resourceId.channelId
})));
// Получаем токен следующей страницы, если он есть
nextPageToken = response.data.nextPageToken;
} while (nextPageToken); // Продолжаем, пока есть токен следующей страницы
return subscriptions;
} catch (error) {
console.error('Ошибка получения подписок', error);
return [];
}
}
Далее функция возвращает ответ и запускается функция, которая проверяет есть ли данный канал в базе данных бота (чтобы в будущем, с увеличением базы, ускорить выдачу каналов).
async function checkAndAddNewChannels(subscriptions, youtubeApiKey, chatId) {
let chat = await Analytics.findOne({ chatId: chatId })
if (chat.awatingChannels) {
let msgWait = await bot.telegram.sendMessage(chatId, `Бот сопоставляет Youtube и Telegram каналы, это займет время...`);
chat.awatingChannels = false
await chat.save()
const youtubeUrls = subscriptions.map(sub => `https://www.youtube.com/channel/${sub.channelId}`);
const foundChannels = await Channel.find({ youtube_url: { $in: youtubeUrls } });
// Фильтрация не найденных каналов
const foundUrls = new Set(foundChannels.map(ch => ch.youtube_url));
const notFoundChannels = subscriptions.filter(sub => !foundUrls.has(`https://www.youtube.com/channel/${sub.channelId}`));
...
Если канал не найдется в базе данных, то запустится функция для парса телеграм каналов из раздела "ссылки" на ютубе. Хотелось бы использовать для этого Youtube API аж 3 версии, но там нет функционала, для получения этих данных ?, поэтому я нашел стороннее API - YouTube operational API (lemnoslife.com), с которым приятно работать. Автору респект! (Единственное, username канала почему-то находится в разделе сообщество, именно в апи лемнос, а дело в том, что не у всех каналов есть вкладка сообщество).
// Функция для получения ссылок из описания канала с помощью Lemson API
async function getChannelLinksFromDescription(channelId) {
try {
const response = await axios.get(`${LEMNOS_API_URL}?part=about&id=${channelId}`);
return response.data;
} catch (error) {
logger.error(`Error getting channel info from Lemnos Life API: ${error}`);
return null;
}
}
Далее новый канал добавляется в базу данных, а после бот выдает несколькими сообщениями все ютуб-каналы, на которые он подписан и через тире их телеграм каналы. В первых версиях, бот рассчитывал только на отправку одного сообщения, но оказалось, что у моего тестового аккаунта подписок сильно больше, чем ограничение на длину сообщения в телеграм. Недолго подумав, я просто решил отправлять всю информацию несколькими сообщениями. (Других вариантов, как будто нет)
Была одна надоедливая проблема, которую я костыльно решил: при авторизации в google аккаунт запрос на сервер отправляется дважды, если пользователь дважды нажимает на выбор аккаунта. Сложно объяснить, но думаю, те кто работал сталкивался с этим. Эта проблема появляется только, если у человека несколько аккаунтов. Поэтому, чтобы сообщения не отправлялись дважды, я стал записывать в документ пользователя в базе данных переменную awatingChannels. При нажатии на кнопку найти каналы ее значение становилось true, а при получение каналов меняло на false, что решило проблему. Думаю, нужно было реализовать это через сессии, что я и пробовал изначально, но что-то пошло не так и я решил по-быстрому все сделать так.
Кнопка: "Связать Youtube-канал с Telegram-каналом" позволяет пользователю вручную добавить связку ютуб-канал + тг-канал, если автор ютуб-канала не оставил ссылку на свой тг в описании. (Сначала запрос отправляется на модерацию мне, а потом, при принятии попадает в общую базу данных).
Для логирования ошибок, впоследствии, стал использовать pino.
Теперь бот работает и выполняет свою функцию. У него есть еще несколько команд, такие как:
/submit (text) - для связи со мной
/faq - ответы на вопросы
/send - чтобы я мог связаться с пользователем
Еще несколько системных и мой прикол - /jericho (иерихон)
bot.command('jericho', async (ctx) => {
if (ctx.message.chat.id == MODERATOR_CHAT_ID) {
ctx.replyWithVideo(`https://i.gifer.com/370.gif`)
throw new Error()
} else {
%*%(НЛО прилетело и скрыло эту надпись здесь)%*%
}
})
Которая вызывает ошибку, а вместе с ней прекрасную гифку для меня: https://i.gifer.com/370.gif
Теперь есть работающий бот, но Google не дает доступ к API тем, про кого не знает (только для тестов). Поэтому нужно пройти проверку. У тебя должен быть сайт, политика, terms и прочее.
Начал я с логотипа и отправил им это
Google сказал, что это лого не отражает вашу индивидуальность. (видимо, из-за слишком явного использования автарок телеграма и ютуба).
Тогда я решил показать всю свою индивидуальность:
Они это приняли, я пошел быстренько сделал сайт, попросил ChatGPT написать политику и отправил. Оказалось надо быть владельцем домена, на котором размещен сайт, поэтому пришлось купить. Подключил сертификаты через certbot. И отправил снова.
На этот раз им не понравилась политика, они жестко ответили, мол: Вы написали, что "Google продает данные пользователей.". Я перечитал все, что ChatGPT написал, не нашел там такого, попросил переделать, отправил снова и теперь уже все приняли :). Еще надо было сделать им демо-видое работы, ну ладно без проблем - FYTT — Найди Ютуб каналы в Телеграмм (Полезный бот в Telegram) | @FYTTproject_bot - YouTube
Потом я решил продвинуть бота и записал два смешных shorts:
https://youtu.be/N0IGLuufSCE - text to speech (elevenlabs)
https://youtu.be/MlXEUIDBhE0 - Speech to speech моей записи на ии оригинального голоса Рика из Рика и Морти (сделал на этом сайте).
Ну и написал первую версии статьи: Я создал Телеграм бота (FYTT), который ищет Телеграм каналы всех ваших подписок на ютубе / Хабр (habr.com)
Получил много положительной реакции, до статьи у меня было 37 человек, которые запустили бота, а теперь 235 из которых 93+ использовали его функционал.
Ссылки:
Бот - https://t.me/FYTTproject_bot
Исходный код - https://github.com/VitoSperansky/FromYoutubeToTelegram
Сайт - fytt.tech:3000 (порт 3000, потому что порты ниже 1024 по умолчанию закрыты для установки серверов, их можно открыть, но это угроза безопасности. По-хорошему мне надо было сделать переадресацию с 443 порта (он дефолтный для https сайтов, поэтому скрыт в поисковой строке и пользователи его не видят) с помощью ngrok например, но я не стал тратить силы, так это все равно заглушка для гугла).
Связаться со мной по вопросам или просто - https://t.me/vitosperansky
jaker
И зачем это?
babypowder
чтобы самолётиком в башенку