Did some refactor, add YouTube as search variant
This commit is contained in:
@@ -0,0 +1 @@
|
|||||||
|
from .search import ServiceSearchFilter
|
||||||
|
|||||||
13
bot/filters/search.py
Normal file
13
bot/filters/search.py
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
from aiogram.filters import BaseFilter
|
||||||
|
from aiogram.types import InlineQuery
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceSearchFilter(BaseFilter):
|
||||||
|
def __init__(self, service_letter: str):
|
||||||
|
self.service_letter = f'{service_letter}:'
|
||||||
|
|
||||||
|
async def __call__(self, inline_query: InlineQuery):
|
||||||
|
return (
|
||||||
|
inline_query.query.startswith(self.service_letter) and
|
||||||
|
inline_query.query != self.service_letter
|
||||||
|
)
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from aiogram import Router
|
from aiogram import Router
|
||||||
from . import (
|
from . import (
|
||||||
initialize,
|
initialize,
|
||||||
|
inline_song,
|
||||||
inline_default,
|
inline_default,
|
||||||
inline_empty,
|
inline_empty,
|
||||||
on_chosen,
|
on_chosen,
|
||||||
@@ -14,6 +15,7 @@ router.chosen_inline_result.outer_middleware(SaveChosenMiddleware())
|
|||||||
|
|
||||||
router.include_routers(
|
router.include_routers(
|
||||||
initialize.router,
|
initialize.router,
|
||||||
|
inline_song.router,
|
||||||
inline_default.router,
|
inline_default.router,
|
||||||
inline_empty.router,
|
inline_empty.router,
|
||||||
on_chosen.router,
|
on_chosen.router,
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ from aiogram import Router, F
|
|||||||
|
|
||||||
from aiogram.types import InlineQuery
|
from aiogram.types import InlineQuery
|
||||||
|
|
||||||
from bot.markups.deezer import get_deezer_search_results
|
from bot.results.deezer import get_deezer_search_results
|
||||||
|
|
||||||
router = Router()
|
router = Router()
|
||||||
|
|
||||||
|
|||||||
10
bot/handlers/inline_song/__init__.py
Normal file
10
bot/handlers/inline_song/__init__.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from aiogram import Router
|
||||||
|
|
||||||
|
from . import on_inline_spotify, on_inline_deezer, on_inline_youtube
|
||||||
|
|
||||||
|
router = Router()
|
||||||
|
router.include_routers(
|
||||||
|
on_inline_spotify.router,
|
||||||
|
on_inline_deezer.router,
|
||||||
|
on_inline_youtube.router,
|
||||||
|
)
|
||||||
17
bot/handlers/inline_song/on_inline_deezer.py
Normal file
17
bot/handlers/inline_song/on_inline_deezer.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
from aiogram import Router
|
||||||
|
|
||||||
|
from aiogram.types import InlineQuery
|
||||||
|
|
||||||
|
from bot.results.deezer import get_deezer_search_results
|
||||||
|
from bot.filters import ServiceSearchFilter
|
||||||
|
|
||||||
|
router = Router()
|
||||||
|
|
||||||
|
|
||||||
|
@router.inline_query(ServiceSearchFilter('d'))
|
||||||
|
async def search_deezer_inline_query(inline_query: InlineQuery):
|
||||||
|
await inline_query.answer(
|
||||||
|
await get_deezer_search_results(inline_query.query.removeprefix('d:')),
|
||||||
|
cache_time=0,
|
||||||
|
is_personal=True
|
||||||
|
)
|
||||||
17
bot/handlers/inline_song/on_inline_spotify.py
Normal file
17
bot/handlers/inline_song/on_inline_spotify.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
from aiogram import Router
|
||||||
|
|
||||||
|
from aiogram.types import InlineQuery
|
||||||
|
|
||||||
|
from bot.results.spotify import get_spotify_search_results
|
||||||
|
from bot.filters import ServiceSearchFilter
|
||||||
|
|
||||||
|
router = Router()
|
||||||
|
|
||||||
|
|
||||||
|
@router.inline_query(ServiceSearchFilter('s'))
|
||||||
|
async def search_spotify_inline_query(inline_query: InlineQuery):
|
||||||
|
await inline_query.answer(
|
||||||
|
await get_spotify_search_results(inline_query.query.removeprefix('s:')),
|
||||||
|
cache_time=0,
|
||||||
|
is_personal=True
|
||||||
|
)
|
||||||
17
bot/handlers/inline_song/on_inline_youtube.py
Normal file
17
bot/handlers/inline_song/on_inline_youtube.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
from aiogram import Router
|
||||||
|
|
||||||
|
from aiogram.types import InlineQuery
|
||||||
|
|
||||||
|
from bot.results.youtube import get_youtube_search_results
|
||||||
|
from bot.filters import ServiceSearchFilter
|
||||||
|
|
||||||
|
router = Router()
|
||||||
|
|
||||||
|
|
||||||
|
@router.inline_query(ServiceSearchFilter('y'))
|
||||||
|
async def search_youtube_inline_query(inline_query: InlineQuery):
|
||||||
|
await inline_query.answer(
|
||||||
|
await get_youtube_search_results(inline_query.query.removeprefix('y:')),
|
||||||
|
cache_time=0,
|
||||||
|
is_personal=True
|
||||||
|
)
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
from aiogram import Router
|
from aiogram import Router
|
||||||
from . import spotify, deezer
|
from . import spotify, deezer, youtube
|
||||||
|
|
||||||
router = Router()
|
router = Router()
|
||||||
|
|
||||||
router.include_routers(
|
router.include_routers(
|
||||||
spotify.router,
|
spotify.router,
|
||||||
deezer.router,
|
deezer.router,
|
||||||
|
youtube.router,
|
||||||
)
|
)
|
||||||
|
|
||||||
__all__ = ['router']
|
__all__ = ['router']
|
||||||
|
|||||||
49
bot/handlers/on_chosen/youtube.py
Normal file
49
bot/handlers/on_chosen/youtube.py
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
from aiogram import Router, Bot, F
|
||||||
|
from aiogram.types import (
|
||||||
|
BufferedInputFile, URLInputFile, InputMediaAudio,
|
||||||
|
ChosenInlineResult,
|
||||||
|
)
|
||||||
|
|
||||||
|
from bot.modules.youtube import youtube, AgeRestrictedError
|
||||||
|
from bot.utils.config import config
|
||||||
|
from bot.modules.database import db
|
||||||
|
|
||||||
|
router = Router()
|
||||||
|
|
||||||
|
|
||||||
|
@router.chosen_inline_result(F.result_id.startswith('yt::'))
|
||||||
|
async def on_new_chosen(chosen_result: ChosenInlineResult, bot: Bot):
|
||||||
|
song = youtube.songs.from_id(chosen_result.result_id.removeprefix('yt::'))
|
||||||
|
|
||||||
|
try:
|
||||||
|
bytestream = await song.to_bytestream()
|
||||||
|
except AgeRestrictedError:
|
||||||
|
await bot.edit_message_caption(
|
||||||
|
inline_message_id=chosen_result.inline_message_id,
|
||||||
|
caption='🔞 This song is age restricted, so I can\'t download it. '
|
||||||
|
'Try downloading it from Deezer or SoundCloud',
|
||||||
|
reply_markup=None
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
audio = await bot.send_audio(
|
||||||
|
chat_id=config.telegram.files_chat,
|
||||||
|
audio=BufferedInputFile(
|
||||||
|
file=bytestream.file,
|
||||||
|
filename=bytestream.filename,
|
||||||
|
),
|
||||||
|
thumbnail=URLInputFile(song.thumbnail),
|
||||||
|
performer=song.all_artists,
|
||||||
|
title=song.name,
|
||||||
|
duration=bytestream.duration,
|
||||||
|
)
|
||||||
|
|
||||||
|
db.spotify[song.id] = audio.audio.file_id
|
||||||
|
|
||||||
|
await bot.edit_message_media(
|
||||||
|
inline_message_id=chosen_result.inline_message_id,
|
||||||
|
media=InputMediaAudio(media=audio.audio.file_id),
|
||||||
|
reply_markup=None
|
||||||
|
)
|
||||||
|
|
||||||
|
await db.occasionally_write()
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
from aiogram.types import (
|
|
||||||
InlineQueryResultDocument, InlineQueryResultCachedAudio,
|
|
||||||
InlineKeyboardMarkup, InlineKeyboardButton, InlineQueryResult
|
|
||||||
)
|
|
||||||
|
|
||||||
from bot.modules.spotify import spotify
|
|
||||||
from bot.modules.database import db
|
|
||||||
|
|
||||||
|
|
||||||
async def get_spotify_search_results(query: str) -> list[
|
|
||||||
InlineQueryResultDocument | InlineQueryResultCachedAudio
|
|
||||||
]:
|
|
||||||
return [
|
|
||||||
InlineQueryResultDocument(
|
|
||||||
id='spot::' + audio.id,
|
|
||||||
title=audio.name,
|
|
||||||
description=audio.all_artists,
|
|
||||||
thumb_url=audio.thumbnail,
|
|
||||||
document_url=audio.preview_url or audio.thumbnail,
|
|
||||||
mime_type='application/zip',
|
|
||||||
reply_markup=InlineKeyboardMarkup(
|
|
||||||
inline_keyboard=[
|
|
||||||
[InlineKeyboardButton(text='Downloading...', callback_data='.')]
|
|
||||||
]
|
|
||||||
),
|
|
||||||
caption=audio.full_name,
|
|
||||||
) if audio.id not in list(db.spotify.keys()) else
|
|
||||||
InlineQueryResultCachedAudio(
|
|
||||||
id='spotc::' + audio.id,
|
|
||||||
audio_file_id=db.spotify[audio.id],
|
|
||||||
)
|
|
||||||
for audio in spotify.songs.search(query, limit=50)
|
|
||||||
]
|
|
||||||
1
bot/modules/common/__init__.py
Normal file
1
bot/modules/common/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
pass
|
||||||
1
bot/modules/common/song/__init__.py
Normal file
1
bot/modules/common/song/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from .song import BaseSongItem
|
||||||
21
bot/modules/common/song/song.py
Normal file
21
bot/modules/common/song/song.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from attrs import define
|
||||||
|
|
||||||
|
|
||||||
|
@define
|
||||||
|
class BaseSongItem:
|
||||||
|
name: str
|
||||||
|
id: str
|
||||||
|
artists: list[str]
|
||||||
|
preview_url: str | None
|
||||||
|
thumbnail: str
|
||||||
|
|
||||||
|
@property
|
||||||
|
def all_artists(self):
|
||||||
|
return ', '.join(self.artists)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def full_name(self):
|
||||||
|
return f"{self.all_artists} - {self.name}"
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.full_name
|
||||||
@@ -17,6 +17,7 @@ class Db(object):
|
|||||||
self.inline = DBDict('inline')
|
self.inline = DBDict('inline')
|
||||||
self.spotify = DBDict('spotify')
|
self.spotify = DBDict('spotify')
|
||||||
self.deezer = DBDict('deezer')
|
self.deezer = DBDict('deezer')
|
||||||
|
self.youtube = DBDict('youtube')
|
||||||
|
|
||||||
async def write(self):
|
async def write(self):
|
||||||
await self.config.write()
|
await self.config.write()
|
||||||
|
|||||||
@@ -2,43 +2,25 @@ from attrs import define
|
|||||||
|
|
||||||
from .driver import DeezerDriver
|
from .driver import DeezerDriver
|
||||||
|
|
||||||
|
from ..common.song import BaseSongItem
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class SongItem:
|
class SongItem(BaseSongItem):
|
||||||
name: str
|
|
||||||
id: int
|
|
||||||
id_s: str
|
|
||||||
artist: str
|
|
||||||
preview_url: str | None
|
|
||||||
thumbnail: str
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_deezer(cls, song_item: dict):
|
def from_deezer(cls, song_item: dict):
|
||||||
return cls(
|
return cls(
|
||||||
name=song_item['title'],
|
name=song_item['title'],
|
||||||
id=song_item['id'],
|
id=str(song_item['id']),
|
||||||
id_s=str(song_item['id']),
|
artists=[song_item['artist']['name']],
|
||||||
artist=song_item['artist']['name'],
|
|
||||||
preview_url=song_item.get('preview'),
|
preview_url=song_item.get('preview'),
|
||||||
thumbnail=song_item['album']['cover_medium']
|
thumbnail=song_item['album']['cover_medium']
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
|
||||||
def full_name(self):
|
|
||||||
return f"{self.artist} - {self.name}"
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.full_name
|
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class FullSongItem:
|
class FullSongItem(BaseSongItem):
|
||||||
name: str
|
|
||||||
id: str
|
|
||||||
artists: list[str]
|
|
||||||
preview_url: str | None
|
|
||||||
duration: int
|
duration: int
|
||||||
thumbnail: str
|
|
||||||
track_dict: dict
|
track_dict: dict
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@@ -60,17 +42,6 @@ class FullSongItem:
|
|||||||
track_dict=song_item
|
track_dict=song_item
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
|
||||||
def all_artists(self):
|
|
||||||
return ', '.join(self.artists)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def full_name(self):
|
|
||||||
return f"{self.all_artists} - {self.name}"
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.full_name
|
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class Songs(object):
|
class Songs(object):
|
||||||
|
|||||||
@@ -1,15 +1,11 @@
|
|||||||
from attrs import define
|
from attrs import define
|
||||||
import spotipy
|
import spotipy
|
||||||
|
|
||||||
|
from ..common.song import BaseSongItem
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class SongItem:
|
class SongItem(BaseSongItem):
|
||||||
name: str
|
|
||||||
id: str
|
|
||||||
artists: list[str]
|
|
||||||
preview_url: str | None
|
|
||||||
thumbnail: str
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_spotify(cls, song_item: dict):
|
def from_spotify(cls, song_item: dict):
|
||||||
return cls(
|
return cls(
|
||||||
@@ -21,17 +17,6 @@ class SongItem:
|
|||||||
thumbnail=song_item['album']['images'][1]['url']
|
thumbnail=song_item['album']['images'][1]['url']
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
|
||||||
def all_artists(self):
|
|
||||||
return ', '.join(self.artists)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def full_name(self):
|
|
||||||
return f"{self.all_artists} - {self.name}"
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return f"{', '.join(self.artists)} - {self.name}"
|
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class Songs(object):
|
class Songs(object):
|
||||||
|
|||||||
@@ -5,13 +5,12 @@ from .downloader import Downloader, YouTubeBytestream
|
|||||||
|
|
||||||
from typing import Awaitable
|
from typing import Awaitable
|
||||||
|
|
||||||
|
from ..common.song import BaseSongItem
|
||||||
|
|
||||||
|
|
||||||
@define
|
@define
|
||||||
class SongItem:
|
class SongItem(BaseSongItem):
|
||||||
name: str
|
preview_url: None = None
|
||||||
id: str
|
|
||||||
artists: list[str]
|
|
||||||
thumbnail: str
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_youtube(cls, song_item: dict):
|
def from_youtube(cls, song_item: dict):
|
||||||
@@ -22,16 +21,14 @@ class SongItem:
|
|||||||
thumbnail=song_item['thumbnails'][1]['url']
|
thumbnail=song_item['thumbnails'][1]['url']
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
@classmethod
|
||||||
def all_artists(self):
|
def from_details(cls, details: dict):
|
||||||
return ', '.join(self.artists)
|
return cls(
|
||||||
|
name=details['title'],
|
||||||
@property
|
id=details['videoId'],
|
||||||
def full_name(self):
|
artists=details['author'].split(' & '),
|
||||||
return f"{self.all_artists} - {self.name}"
|
thumbnail=details['thumbnail']['thumbnails'][1]['url']
|
||||||
|
)
|
||||||
def __str__(self):
|
|
||||||
return self.full_name
|
|
||||||
|
|
||||||
def to_bytestream(self) -> Awaitable[YouTubeBytestream]:
|
def to_bytestream(self) -> Awaitable[YouTubeBytestream]:
|
||||||
return Downloader.from_id(self.id).to_bytestream()
|
return Downloader.from_id(self.id).to_bytestream()
|
||||||
@@ -58,4 +55,4 @@ class Songs(object):
|
|||||||
if r is None:
|
if r is None:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
return SongItem.from_youtube(r)
|
return SongItem.from_details(r['videoDetails'])
|
||||||
|
|||||||
1
bot/results/__init__.py
Normal file
1
bot/results/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
pass
|
||||||
1
bot/results/common/__init__.py
Normal file
1
bot/results/common/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
pass
|
||||||
@@ -3,18 +3,25 @@ from aiogram.types import (
|
|||||||
InlineKeyboardMarkup, InlineKeyboardButton
|
InlineKeyboardMarkup, InlineKeyboardButton
|
||||||
)
|
)
|
||||||
|
|
||||||
from bot.modules.deezer import deezer
|
from bot.modules.database.db import DBDict
|
||||||
from bot.modules.database import db
|
|
||||||
|
from bot.modules.common.song import BaseSongItem
|
||||||
|
from typing import TypeVar
|
||||||
|
|
||||||
|
|
||||||
async def get_deezer_search_results(query: str) -> list[
|
BaseSongT = TypeVar('BaseSongT', bound=BaseSongItem)
|
||||||
InlineQueryResultDocument | InlineQueryResultCachedAudio
|
|
||||||
]:
|
|
||||||
return [
|
async def get_common_search_result(
|
||||||
|
audio: BaseSongT,
|
||||||
|
db_table: DBDict,
|
||||||
|
service_id: str
|
||||||
|
) -> InlineQueryResultDocument | InlineQueryResultCachedAudio:
|
||||||
|
return (
|
||||||
InlineQueryResultDocument(
|
InlineQueryResultDocument(
|
||||||
id='deez::' + audio.id_s,
|
id=f'{service_id}::' + audio.id,
|
||||||
title=audio.name,
|
title=audio.name,
|
||||||
description=audio.artist,
|
description=audio.all_artists,
|
||||||
thumb_url=audio.thumbnail,
|
thumb_url=audio.thumbnail,
|
||||||
document_url=audio.preview_url or audio.thumbnail,
|
document_url=audio.preview_url or audio.thumbnail,
|
||||||
mime_type='application/zip',
|
mime_type='application/zip',
|
||||||
@@ -24,10 +31,9 @@ async def get_deezer_search_results(query: str) -> list[
|
|||||||
]
|
]
|
||||||
),
|
),
|
||||||
caption=audio.full_name,
|
caption=audio.full_name,
|
||||||
) if audio.id_s not in list(db.deezer.keys()) else
|
) if audio.id not in list(db_table.keys()) else
|
||||||
InlineQueryResultCachedAudio(
|
InlineQueryResultCachedAudio(
|
||||||
id='deezc::' + audio.id_s,
|
id=f'{service_id}c::' + audio.id,
|
||||||
audio_file_id=db.deezer[audio.id_s],
|
audio_file_id=db_table[audio.id],
|
||||||
)
|
)
|
||||||
for audio in await deezer.songs.search(query, limit=50)
|
)
|
||||||
]
|
|
||||||
21
bot/results/deezer/search.py
Normal file
21
bot/results/deezer/search.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from aiogram.types import (
|
||||||
|
InlineQueryResultDocument, InlineQueryResultCachedAudio
|
||||||
|
)
|
||||||
|
|
||||||
|
from bot.modules.deezer import deezer
|
||||||
|
from bot.modules.database import db
|
||||||
|
|
||||||
|
from ..common.search import get_common_search_result
|
||||||
|
|
||||||
|
|
||||||
|
async def get_deezer_search_results(query: str) -> list[
|
||||||
|
InlineQueryResultDocument | InlineQueryResultCachedAudio
|
||||||
|
]:
|
||||||
|
return [
|
||||||
|
await get_common_search_result(
|
||||||
|
audio=audio,
|
||||||
|
db_table=db.deezer,
|
||||||
|
service_id='deez'
|
||||||
|
)
|
||||||
|
for audio in await deezer.songs.search(query, limit=50)
|
||||||
|
]
|
||||||
21
bot/results/spotify/search.py
Normal file
21
bot/results/spotify/search.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from aiogram.types import (
|
||||||
|
InlineQueryResultDocument, InlineQueryResultCachedAudio
|
||||||
|
)
|
||||||
|
|
||||||
|
from bot.modules.spotify import spotify
|
||||||
|
from bot.modules.database import db
|
||||||
|
|
||||||
|
from ..common.search import get_common_search_result
|
||||||
|
|
||||||
|
|
||||||
|
async def get_spotify_search_results(query: str) -> list[
|
||||||
|
InlineQueryResultDocument | InlineQueryResultCachedAudio
|
||||||
|
]:
|
||||||
|
return [
|
||||||
|
await get_common_search_result(
|
||||||
|
audio=audio,
|
||||||
|
db_table=db.spotify,
|
||||||
|
service_id='spot'
|
||||||
|
)
|
||||||
|
for audio in spotify.songs.search(query, limit=50)
|
||||||
|
]
|
||||||
6
bot/results/youtube/__init__.py
Normal file
6
bot/results/youtube/__init__.py
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
from .search import get_youtube_search_results
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
'get_youtube_search_results'
|
||||||
|
]
|
||||||
21
bot/results/youtube/search.py
Normal file
21
bot/results/youtube/search.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from aiogram.types import (
|
||||||
|
InlineQueryResultDocument, InlineQueryResultCachedAudio
|
||||||
|
)
|
||||||
|
|
||||||
|
from bot.modules.youtube import youtube
|
||||||
|
from bot.modules.database import db
|
||||||
|
|
||||||
|
from ..common.search import get_common_search_result
|
||||||
|
|
||||||
|
|
||||||
|
async def get_youtube_search_results(query: str) -> list[
|
||||||
|
InlineQueryResultDocument | InlineQueryResultCachedAudio
|
||||||
|
]:
|
||||||
|
return [
|
||||||
|
await get_common_search_result(
|
||||||
|
audio=audio,
|
||||||
|
db_table=db.youtube,
|
||||||
|
service_id='yt'
|
||||||
|
)
|
||||||
|
for audio in youtube.songs.search(query, limit=40)
|
||||||
|
]
|
||||||
Reference in New Issue
Block a user