Merge pull request #12 from Kenearos/claude/init-irc-bot-dashboard-kYxih

Claude/init irc bot dashboard k yxih
This commit is contained in:
Kenearos 2026-01-02 22:42:31 +01:00 committed by GitHub
commit 952e543970
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 133 additions and 15 deletions

View file

@ -111,6 +111,62 @@ class EugenBot(irc.bot.SingleServerIRCBot):
self.handle_mention(username, message),
self.loop
)
# Check for ambiguous greetings (AI will decide if bot is addressed)
elif self.detector.is_ambiguous_greeting(message):
self.logger.debug(f"Ambiguous greeting detected from {username}: {message}")
if self.dashboard:
self.dashboard.log_event("info", {"message": f"Ambiguous greeting: {message}"})
# Ask AI if bot is addressed
asyncio.run_coroutine_threadsafe(
self.check_if_addressed(username, message),
self.loop
)
async def check_if_addressed(self, username, message):
"""
Check with AI if an ambiguous message is addressed to the bot
Args:
username (str): User who sent the message
message (str): Ambiguous message (e.g., "Hi wie gehts")
"""
try:
# Ask AI if the message is directed at the bot
check_prompt = f"""You are {self.bot_name}, a Twitch chat bot.
A user named {username} just wrote: "{message}"
This message doesn't explicitly mention you, but it might be directed at you.
Consider:
- Is this a greeting or question that could be for the bot?
- Is there recent conversation context suggesting it's for you?
- Or is it likely a general chat message not for the bot?
Respond with ONLY "YES" if the message is likely addressed to you, or "NO" if not.
Do not explain, just answer YES or NO."""
messages = [{"role": "user", "content": check_prompt}]
self.logger.debug(f"Asking AI if addressed: {message}")
response = await self.ai.get_response(messages)
if response and "YES" in response.upper():
self.logger.debug(f"AI says message is for bot: {message}")
if self.dashboard:
self.dashboard.log_event("info", {"message": f"AI confirmed: message is for bot"})
# Treat as mention and respond
await self.handle_mention(username, message)
else:
self.logger.debug(f"AI says message is not for bot: {message}")
if self.dashboard:
self.dashboard.log_event("info", {"message": f"AI confirmed: message is not for bot"})
except Exception as e:
self.logger.error(f"Error checking if addressed: {str(e)}")
if self.dashboard:
self.dashboard.log_event("error", {"error": str(e)})
async def handle_mention(self, username, message):
"""

View file

@ -13,20 +13,53 @@ class MentionDetector:
def __init__(self, bot_name="Eugen"):
self.bot_name = bot_name
# Generate nicknames/partial names from bot_name
self.nicknames = self._generate_nicknames(bot_name)
# Create patterns for various mention formats
self.patterns = [
rf"@{bot_name}", # @Eugen
rf"{bot_name}:", # Eugen:
rf"{bot_name},", # Eugen,
rf"^{bot_name}\s", # Eugen at start
rf"\s{bot_name}\s", # Eugen in middle
rf"\s{bot_name}$", # Eugen at end
]
# Include bot name and all nicknames
all_names = [bot_name] + self.nicknames
self.patterns = []
for name in all_names:
self.patterns.extend([
rf"@{name}\b", # @name (with word boundary)
rf"\b{name}[:!?.,]", # name: name! name? name, name.
rf"^{name}\b", # name at start of message
rf"\b{name}\b", # name anywhere as whole word
])
# Case-insensitive compilation
self.compiled_patterns = [
re.compile(pattern, re.IGNORECASE) for pattern in self.patterns
]
# Patterns for ambiguous greetings (might be directed at bot)
self.greeting_patterns = [
r"^(hi|hey|hallo|hello|servus|moin)(\s|$|\W)",
r"^(wie\s+geht'?s|wie\s+gehts|how\s+are\s+you)",
r"^(alles\s+klar|everything\s+ok)",
]
self.compiled_greetings = [
re.compile(pattern, re.IGNORECASE) for pattern in self.greeting_patterns
]
def _generate_nicknames(self, bot_name):
"""Generate common nicknames from bot name"""
nicknames = []
# For kenearosmd, generate: Kene, Kenearos
if len(bot_name) >= 4:
nicknames.append(bot_name[:4]) # First 4 chars (Kene)
if len(bot_name) >= 8:
nicknames.append(bot_name[:8]) # First 8 chars (Kenearos)
# Remove duplicates and the full name
nicknames = [n for n in set(nicknames) if n != bot_name]
return nicknames
def is_mentioned(self, message):
"""
Check if bot was mentioned in message
@ -45,6 +78,28 @@ class MentionDetector:
return True
return False
def is_ambiguous_greeting(self, message):
"""
Check if message is an ambiguous greeting that might be for the bot
Args:
message (str): Chat message to check
Returns:
bool: True if message is an ambiguous greeting
"""
if not message:
return False
# Don't check if there's already a clear mention
if self.is_mentioned(message):
return False
for pattern in self.compiled_greetings:
if pattern.search(message):
return True
return False
def extract_content(self, message):
"""
Extract message content without the mention
@ -58,15 +113,22 @@ class MentionDetector:
if not message:
return ""
# Remove common mention patterns
# Remove bot name and nickname mentions from the message
content = message
patterns_to_remove = [
rf"@{self.bot_name}[,:]?\s*",
rf"{self.bot_name}[,:]?\s*",
]
all_names = [self.bot_name] + self.nicknames
for pattern in patterns_to_remove:
content = re.sub(pattern, "", content, flags=re.IGNORECASE)
for name in all_names:
# Remove @mention at start
content = re.sub(rf"^@{name}\b[,:]?\s*", "", content, flags=re.IGNORECASE)
# Remove name at start with optional punctuation
content = re.sub(rf"^{name}\b[,:]?\s*", "", content, flags=re.IGNORECASE)
# Remove name at end with optional punctuation
content = re.sub(rf"\s*\b{name}[,!?.]?\s*$", "", content, flags=re.IGNORECASE)
# Remove name in middle with punctuation
content = re.sub(rf"\s*\b{name}[,:!?]\s*", " ", content, flags=re.IGNORECASE)
return content.strip()