# { # 'speciesCode': 'dowwoo', # 'comName': 'Downy Woodpecker', # 'sciName': 'Dryobates pubescens', # 'locId': 'L36986367', # 'locName': 'Home', # 'obsDt': '2024-12-14 17:06', # 'howMany': 1, # 'lat': 36.0006572, # 'lng': -95.0865753, # 'obsValid': True, # 'obsReviewed': False, # 'locationPrivate': True, # 'subId': 'S205413945' # } from random import choice import requests from time import sleep import socket import re from json import dump, load, dumps try: with open("config.json", "r") as f: config = load(f) except FileNotFoundError: exit("Please create config.json with api key(s) and channels") host = "localhost" port = 6667 nick = config["nick"] realname = "a bot by ~nebula" helptext = "!birds, !trivia, !trscores, !aitrivia, !aiscores. contact ~nebula for help, feedback or problem reports. https://git.tilde.town/nebula/mysterious_cube" channels = config["channels"] channel_re = re.compile(r"PRIVMSG (#*\w+)") name_re = re.compile(r"^:([^!]*)!") llm_answer_re = re.compile(r"^(true|false)") llama_url = "https://llama.mcopp.com/v1/chat/completions" llama_headers = { "Content-Type": "application/json", "Authorization": config["llama_key"] } geonames_url = "http://api.geonames.org/searchJSON" geonames_user = config["geonames_user"] ebird_url = "https://api.ebird.org/v2/data/obs/geo/recent" ebird_key = config["ebird_key"] google_url = "https://customsearch.googleapis.com/customsearch/v1" google_key = config["google_key"] google_cx = config["google_cx"] trivia_questions_file = "trivia.questions" trivia_state_file = "trivia.state" trivia_score_file = "trivia.scores" trivia_unselected_file = "trivia.unselected" ai_state_file = "trivia.aistate" ai_score_file = "trivia.aiscores" bird_url_file = "brids.urls" try: with open(trivia_questions_file, "r") as f: trivia_questions = load(f) except FileNotFoundError: trivia_questions = [] try: with open(trivia_state_file, "r") as f: trivia_state = load(f) except FileNotFoundError: trivia_state = {} try: with open(trivia_score_file, "r") as f: trivia_scores = load(f) except FileNotFoundError: trivia_scores = {} try: with open(trivia_unselected_file, "r") as f: trivia_unselected = load(f) except FileNotFoundError: trivia_unselected = [] try: with open(ai_state_file, "r") as f: ai_state = load(f) except FileNotFoundError: ai_state = {} try: with open(ai_score_file, "r") as f: ai_scores = load(f) except FileNotFoundError: ai_scores = {} try: with open(bird_url_file, "r") as f: bird_urls = load(f) except FileNotFoundError: bird_urls = {} def write_state(): with open(trivia_state_file, "w") as f: dump(trivia_state, f) with open(trivia_score_file, "w") as f: dump(trivia_scores, f) with open(trivia_unselected_file, "w") as f: dump(trivia_unselected, f) with open(ai_score_file, "w") as f: dump(ai_scores, f) with open(ai_state_file, "w") as f: dump(ai_state, f) with open(bird_url_file, "w") as f: dump(bird_urls, f) def get_location(query): params = { "username": geonames_user, "q": query, "maxRows": 1 } try: response = requests.get(geonames_url, params=params) data = response.json()["geonames"][0] if "lat" not in data: return None else: return data except IndexError: return None def cache_bird_url(sciName): global bird_urls if sciName in bird_urls.keys(): return bird_urls[sciName] search = requests.get(google_url, params={ "q": sciName, "key": google_key, "cx": google_cx }) data = search.json() try: link = data["items"][0]["link"] bird_urls[sciName] = link write_state() return link except (IndexError, KeyError): return None def get_birds(location): params = { "key": ebird_key, "dist": 50, "lat": location["lat"], "lng": location["lng"] } request = requests.get(ebird_url, params=params) data = request.json()[:4] line = f"Location: {location['name']}; " for sighting in data: url = cache_bird_url(sighting['sciName']).rstrip("/id") line += f"{sighting['comName']} [ {url} ]; " return line.rstrip("; ") def post_birds(channel, username, arguments): if not arguments: return "Posts recently sighted birds. Give a location name with this command, eg !birds Dodge City, KS" location = get_location(arguments) if not location: return f"No data found for {arguments}" birds = get_birds(location) if not birds: return f"No data found for {arguments}" return birds def get_question(ai_enabled=False): global trivia_questions global trivia_unselected if trivia_questions: if not trivia_unselected: trivia_unselected = trivia_questions.copy() question = choice(trivia_unselected) trivia_unselected.remove(question) question.append(ai_enabled) # print(len(unselected)) return question else: return False def post_question(channel, username, arguments): global trivia_state question = get_question() if question: trivia_state[channel] = question write_state() return f"Answer 'true' or 'false': {question[0]}" else: return "internal error" def post_ai_question(channel, username, arguments): global ai_state question = get_question(ai_enabled=True) if question: ai_state[channel] = question write_state() return f"Will AI answer this true/false statement 'right' or 'wrong': {question[0]}" else: return "internal error" def ai_answer(choice, channel, name): global ai_state if channel not in ai_state.keys(): return None question_text , answer, ai_enabled = ai_state[channel] user_correct = False try: llm_response = llama_response(question_text) llm_answer = llm_answer_re.search(llm_response.lower()) except Exception as e: print(e) return "internal error" del ai_state[channel] write_state() if llm_answer: llm_answer = llm_answer.group(1) if llm_answer.lower() == answer: line = "The AI was (at least kind of) right! " user_correct = choice == "right" else: line = "The AI was wrong! " user_correct = choice == "wrong" else: return [ f"Cannot automatically determine if AI is right or wrong.", f"AI Response: {llm_response}", f"The right answer is {answer}!" ] # print(f"{answer}; {choice}; {user_correct}") if name: if name not in ai_scores.keys(): ai_scores[name] = 0 if user_correct: ai_scores[name] += 1 write_state() return [ f"AI response: {llm_response}", line + f"{name} scores 1 AI point! Total AI score for {name}: {ai_scores[name]}pts. See top AI scores with !aiscores" ] else: ai_scores[name] -= 1 write_state() return [ f"AI response: {llm_response}", line + f"{name} loses 1 AI point! Total AI score for {name}: {ai_scores[name]}pts. See top AI scores with !aiscores" ] return [ f"AI response: {llm_response}", f"The right answer is {answer}!" ] def answer(choice, channel, name): global trivia_state if channel not in trivia_state.keys(): return None _, answer, ai_enabled = trivia_state[channel] del trivia_state[channel] write_state() line = f"The answer is {answer}!" if not ai_enabled and name: if name not in trivia_scores.keys(): trivia_scores[name] = 0 if choice == answer: trivia_scores[name] += 1 line += f" {name} scores 1 point! Total score for {name}: {trivia_scores[name]}pts." else: trivia_scores[name] -= 1 line += f" {name} loses 1 point! Total score for {name}: {trivia_scores[name]}pts." write_state() line += " See top scores with !trscores" return line def post_top_scores(channel, name, arguments): global trivia_scores score_list = [(name, score) for name, score in trivia_scores.items()] if not score_list: return "No current scores." sorted_scores = sorted(score_list, key=lambda x: x[1], reverse=True) line = "Top scores: " count = 1 for name, score in sorted_scores: if count > 10: break line += f"[{count}. {make_no_ping_username(name)}: {score}pts], " count += 1 return line[:-2] def post_top_ai_scores(channel, name, arguments): global ai_scores score_list = [(name, score) for name, score in ai_scores.items()] if not score_list: return "No current AI scores." sorted_scores = sorted(score_list, key=lambda x: x[1], reverse=True) line = "Top AI scores: " count = 1 for name, score in sorted_scores: if count > 10: break line += f"[{count}. {make_no_ping_username(name)}: {score}pts], " count += 1 return line[:-2] def post_help(channel, name, arguments): return helptext def answer_true(channel, name, arguments): return answer("true", channel, name) def answer_false(channel, name, arguments): return answer("false", channel, name) def answer_right(channel, name, arguments): return ai_answer("right", channel, name) def answer_wrong(channel, name, arguments): return ai_answer("wrong", channel, name) def make_no_ping_username(name): return name[0] + "\u200b" + name[1:] def llama_response(question): content = { "n_predict": 64, "temperature": 0.6, "min_p": 0.05, "messages": [ { "role": "system", "content": "You are an entertaining bot in an IRC server. Your responses are brief." }, { "role": "user", "content": f"{question} True, or false? Briefly explain why." } ] } r = requests.post(llama_url, headers=llama_headers, data=dumps(content)) response = r.json() return response["choices"][0]["message"]["content"] class IRCBot(): def __init__(self, nick, realname, helptext, commands, searchers, channels): self.s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.s.connect((host, port)) self.nick = nick self.realname = realname self.helptext = helptext self.commands = commands self.searchers = searchers self.channels = channels self.sendline(f"NICK {self.nick}") self.sendline(f"USER {self.nick} 0 * :{self.realname}") for channel in self.channels: self.sendline(f"JOIN {channel}") def sendline(self, line): if line: return self.s.send(bytes(f"{line}\r\n", "UTF-8")) return None def send(self, channel, content): if isinstance(content, list): for line in content: self.sendline(f"PRIVMSG {channel} :{line}") sleep(0.5) elif isinstance(content, str): self.sendline(f"PRIVMSG {channel} :{content}") def command_loop(self): while True: char = self.s.recv(1) if not char: exit(f"{self.nick}: no response from IRC server") line = b"" while char != b"\n": if char != b"\r": line += char char = self.s.recv(1) line = line.decode("UTF-8").strip() if line.startswith("PING"): pong = "PONG " + line[5:] self.sendline(pong) continue channel_search = channel_re.search(line) if not channel_search: continue channel = channel_search.group(1) name_search = name_re.search(line) if name_search: name = name_search.group(1) else: name = None if name and not channel.startswith("#"): channel = name try: message_body = line[line.index(" :") + 2:] except (IndexError, ValueError): message_body = "" if message_body: for callback in self.searchers: result = callback(message_body) if result: self.send(channel, result) for command, callback in self.commands: if message_body.lower().startswith(command): try: arguments = line[line.index(command) + len(command) + 1:] except (IndexError, ValueError): arguments = None result = callback(channel, name, arguments) if result: self.send(channel, result) def run(): bot = IRCBot( nick, realname, helptext, [ # endswith commands ("!help", post_help), ("!rollcall", post_help), ("!birds", post_birds), ("!trivia", post_question), ("!aitrivia", post_ai_question), ("!trscores", post_top_scores), ("!aiscores", post_top_ai_scores), ("true", answer_true), ("false", answer_false), ("right", answer_right), ("wrong", answer_wrong) ], [ # message searchers # empty ], channels ) while True: sleep(0.5) bot.command_loop() if __name__ == "__main__": run()