seawolf2357's picture
Update app.py
4f7204d verified
raw
history blame
5.81 kB
import discord
import logging
import os
from huggingface_hub import InferenceClient
import requests
import asyncio
from googleapiclient.discovery import build
from dotenv import load_dotenv
# ν™˜κ²½ λ³€μˆ˜ λ‘œλ“œ
load_dotenv()
# λ‘œκΉ… μ„€μ •
logging.basicConfig(level=logging.DEBUG, format='%(asctime)s:%(levelname)s:%(name)s:%(message)s', handlers=[logging.StreamHandler()])
# μΈν…νŠΈ μ„€μ •
intents = discord.Intents.default()
intents.message_content = True
intents.messages = True
intents.guilds = True
intents.guild_messages = True
# μΆ”λ‘  API ν΄λΌμ΄μ–ΈνŠΈ μ„€μ •
hf_client = InferenceClient("CohereForAI/c4ai-command-r-plus", token=os.getenv("HF_TOKEN"))
# Pexels API ν‚€ μ„€μ •
PEXELS_API_KEY = os.getenv("PEXELS_API_KEY")
PEXELS_VIDEO_API_URL = "https://api.pexels.com/videos/search"
# YouTube API ν‚€ μ„€μ •
YOUTUBE_API_KEY = os.getenv("YOUTUBE_API_KEY")
youtube_service = build('youtube', 'v3', developerKey=YOUTUBE_API_KEY)
# νŠΉμ • 채널 ID
SPECIFIC_CHANNEL_ID = int(os.getenv("DISCORD_CHANNEL_ID"))
YOUTUBE_MUSIC_CHANNEL_ID = os.getenv("YOUTUBE_MUSIC_CHANNEL_ID")
# λŒ€ν™” νžˆμŠ€ν† λ¦¬λ₯Ό μ €μž₯ν•  μ „μ—­ λ³€μˆ˜
conversation_history = []
class MyClient(discord.Client):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.is_processing = False
async def on_ready(self):
logging.info(f'{self.user}둜 λ‘œκ·ΈμΈλ˜μ—ˆμŠ΅λ‹ˆλ‹€!')
# 봇이 μ‹œμž‘λ  λ•Œ μ•ˆλ‚΄ λ©”μ‹œμ§€λ₯Ό 전솑
channel = self.get_channel(SPECIFIC_CHANNEL_ID)
if channel:
await channel.send("찾고싢은 λΉ„λ””μ˜€λ‚˜ μŒμ•…μ— λŒ€ν•œ μ„€λͺ…을 ν•œ λ¬Έμž₯ λ‹¨μœ„λ‘œ μž…λ ₯ν•˜μ„Έμš”. 예) λ°”λ‹€μ—μ„œ μ„œν•‘ν•˜λŠ” μ‚¬λžŒ λ˜λŠ” ν₯겨운 여름 μŒμ•…")
async def on_message(self, message):
if message.author == self.user:
return
if not self.is_message_in_specific_channel(message):
return
if self.is_processing:
return
self.is_processing = True
try:
# 의미 λΆ„μ„ν•˜μ—¬ 영문 ν‚€μ›Œλ“œ μΆ”μΆœ
keywords = await extract_keywords(message)
if keywords:
# Pexels API둜 고해상도 λΉ„λ””μ˜€ 검색
video_urls = await search_videos(keywords)
# YouTube API둜 μŒμ•… 검색
music_urls = await search_music_on_youtube(keywords)
if video_urls or music_urls:
# μš”μ²­μžμ™€μ˜ μ“°λ ˆλ“œ 생성 및 κ²°κ³Ό 전솑
await create_thread_and_send_results(message, keywords, video_urls, music_urls)
else:
await message.channel.send(f"**{keywords}**에 λŒ€ν•œ λΉ„λ””μ˜€λ‚˜ μŒμ•…μ„ 찾을 수 μ—†μŠ΅λ‹ˆλ‹€.")
else:
await message.channel.send("ν‚€μ›Œλ“œλ₯Ό μΆ”μΆœν•  수 μ—†μŠ΅λ‹ˆλ‹€.")
finally:
self.is_processing = False
def is_message_in_specific_channel(self, message):
# λ©”μ‹œμ§€κ°€ μ§€μ •λœ μ±„λ„μ΄κ±°λ‚˜, ν•΄λ‹Ή μ±„λ„μ˜ μ“°λ ˆλ“œμΈ 경우 True λ°˜ν™˜
return message.channel.id == SPECIFIC_CHANNEL_ID or (
isinstance(message.channel, discord.Thread) and message.channel.parent_id == SPECIFIC_CHANNEL_ID
)
async def extract_keywords(message):
user_input = message.content
system_prompt = "λ‹€μŒ λ¬Έμž₯의 μ˜λ―Έμ— λ§žλŠ” 영문 ν‚€μ›Œλ“œλ₯Ό μΆ”μΆœν•˜μ„Έμš”: "
logging.debug(f'Extracting keywords from user input: {user_input}')
messages = [{"role": "system", "content": system_prompt + user_input}]
logging.debug(f'Messages to be sent to the model: {messages}')
loop = asyncio.get_event_loop()
response = await loop.run_in_executor(None, lambda: hf_client.chat_completion(
messages, max_tokens=10, temperature=0.7, top_p=0.85))
# Hugging Face 응닡 νŒŒμ‹±
if response.choices and response.choices[0].message:
keywords = response.choices[0].message['content'].strip()
else:
keywords = ""
logging.debug(f'Extracted keywords: {keywords}')
return keywords
async def search_videos(keywords):
headers = {
"Authorization": PEXELS_API_KEY
}
params = {
"query": keywords,
"per_page": 10 # μ΅œλŒ€ 10개 λΉ„λ””μ˜€λ₯Ό κ°€μ Έμ˜΅λ‹ˆλ‹€
}
response = requests.get(PEXELS_VIDEO_API_URL, headers=headers, params=params)
if response.status_code == 200:
data = response.json()
return [video['video_files'][0]['link'] for video in data['videos']]
return None
async def search_music_on_youtube(keywords):
response = youtube_service.search().list(
q=keywords,
part='snippet',
type='video',
maxResults=5,
channelId=YOUTUBE_MUSIC_CHANNEL_ID # νŠΉμ • 채널 ID둜 검색
).execute()
music_urls = []
for item in response.get('items', []):
music_urls.append(f"https://www.youtube.com/watch?v={item['id']['videoId']}")
return music_urls
async def create_thread_and_send_results(message, keywords, video_urls, music_urls):
# μ“°λ ˆλ“œ 생성
thread = await message.channel.create_thread(name=f"{message.author.name}의 검색 κ²°κ³Ό", message=message)
if video_urls:
message_content = f"**{keywords}**에 λŒ€ν•œ 고해상도 λΉ„λ””μ˜€ {len(video_urls)}개λ₯Ό μ°Ύμ•˜μŠ΅λ‹ˆλ‹€:"
await thread.send(message_content)
for url in video_urls:
await thread.send(url)
if music_urls:
message_content = f"**{keywords}**에 λŒ€ν•œ μŒμ•… {len(music_urls)}개λ₯Ό μ°Ύμ•˜μŠ΅λ‹ˆλ‹€:"
await thread.send(message_content)
for url in music_urls:
await thread.send(url)
if __name__ == "__main__":
discord_client = MyClient(intents=intents)
discord_client.run(os.getenv('DISCORD_TOKEN'))