### # Copyright (c) 2023, oddluck # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # * Redistributions of source code must retain the above copyright notice, # this list of conditions, and the following disclaimer. # * Redistributions in binary form must reproduce the above copyright notice, # this list of conditions, and the following disclaimer in the # documentation and/or other materials provided with the distribution. # * Neither the name of the author of this software nor the name of # contributors to this software may be used to endorse or promote products # derived from this software without specific prior written consent. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. ### from supybot import utils, plugins, ircutils, callbacks from supybot.commands import * from supybot.i18n import PluginInternationalization import re #import openai from openai import OpenAI _ = PluginInternationalization("ChatGPT") class ChatGPT(callbacks.Plugin): """Use the OpenAI ChatGPT API""" threaded = True def __init__(self, irc): self.__parent = super(ChatGPT, self) self.__parent.__init__(irc) self.history = {} def chat(self, irc, msg, args, text): """Manual Call to the ChatGPT API""" channel = msg.channel if not irc.isChannel(channel): channel = msg.nick if not self.registryValue("enabled", msg.channel): return if self.registryValue("nick_include", msg.channel): text = "%s: %s" % (msg.nick, text) # Initialize client client = OpenAI( api_key=self.registryValue("api_key"), base_url=self.registryValue("base_url") ) self.history.setdefault(channel, None) max_history = self.registryValue("max_history", msg.channel) prompt = self.registryValue("prompt", msg.channel).replace("$botnick", irc.nick) if not self.history[channel] or max_history < 1: self.history[channel] = [] model_name = self.registryValue("model", msg.channel) # Base request parameters request_params = { "model": model_name, "messages": self.history[channel][-max_history:] + [ {"role": "system", "content": prompt}, {"role": "user", "content": text} ], "temperature": self.registryValue("temperature", msg.channel), "top_p": self.registryValue("top_p", msg.channel), "max_tokens": self.registryValue("max_tokens", msg.channel), "presence_penalty": self.registryValue("presence_penalty", msg.channel), "user": msg.nick, } # Gemini models fail if frequency_penalty is included if "gemini" not in model_name.lower(): request_params["frequency_penalty"] = self.registryValue("frequency_penalty", msg.channel) completion = client.chat.completions.create(**request_params) if self.registryValue("nick_strip", msg.channel): content = re.sub( r"^%s: " % (irc.nick), "", completion.choices[0].message.content ) else: content = completion.choices[0].message.content prefix = self.registryValue("nick_prefix", msg.channel) if self.registryValue("reply_intact", msg.channel): for line in content.splitlines(): if line: irc.reply(line, prefixNick=prefix) else: response = " ".join(content.splitlines()) irc.reply(response, prefixNick=prefix) self.history[channel].append({"role": "user", "content": text}) self.history[channel].append({"role": "assistant", "content": content}) chat = wrap(chat, ["text"]) Class = ChatGPT # vim:set shiftwidth=4 softtabstop=4 expandtab textwidth=79: