parent
681ad81358
commit
ae6d0532bd
6
Makefile
6
Makefile
|
@ -1,4 +1,4 @@
|
|||
docker-push:
|
||||
docker build . -t matrix-bots
|
||||
docker tag matrix-bots registry.ewintr.nl/matrix-gptzoo
|
||||
docker push registry.ewintr.nl/matrix-gptzoo
|
||||
docker build . -t matrix-gptzoo:latest
|
||||
docker tag matrix-gptzoo:latest registry.ewintr.nl/matrix-gptzoo:latest
|
||||
docker push registry.ewintr.nl/matrix-gptzoo:latest
|
|
@ -0,0 +1,202 @@
|
|||
package bot
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/chzyer/readline"
|
||||
"github.com/rs/zerolog"
|
||||
"github.com/sashabaranov/go-openai"
|
||||
"golang.org/x/exp/slog"
|
||||
"maunium.net/go/mautrix"
|
||||
"maunium.net/go/mautrix/crypto/cryptohelper"
|
||||
"maunium.net/go/mautrix/event"
|
||||
"maunium.net/go/mautrix/format"
|
||||
"maunium.net/go/mautrix/id"
|
||||
)
|
||||
|
||||
type Config struct {
|
||||
Homeserver string
|
||||
UserID string
|
||||
UserAccessKey string
|
||||
UserPassword string
|
||||
UserDisplayName string
|
||||
DBPath string
|
||||
Pickle string
|
||||
OpenAIKey string
|
||||
}
|
||||
|
||||
type Bot struct {
|
||||
config Config
|
||||
readline *readline.Instance
|
||||
client *mautrix.Client
|
||||
cryptoHelper *cryptohelper.CryptoHelper
|
||||
characters []Character
|
||||
conversations Conversations
|
||||
gptClient *GPT
|
||||
logger *slog.Logger
|
||||
}
|
||||
|
||||
func New(cfg Config, logger *slog.Logger) *Bot {
|
||||
return &Bot{
|
||||
config: cfg,
|
||||
logger: logger,
|
||||
}
|
||||
}
|
||||
|
||||
func (m *Bot) Init() error {
|
||||
client, err := mautrix.NewClient(m.config.Homeserver, id.UserID(m.config.UserID), m.config.UserAccessKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
var oei mautrix.OldEventIgnorer
|
||||
oei.Register(client.Syncer.(mautrix.ExtensibleSyncer))
|
||||
m.client = client
|
||||
m.client.Log = zerolog.New(zerolog.NewConsoleWriter(func(w *zerolog.ConsoleWriter) {
|
||||
w.TimeFormat = time.Stamp
|
||||
})).With().Timestamp().Logger().Level(zerolog.InfoLevel)
|
||||
|
||||
m.cryptoHelper, err = cryptohelper.NewCryptoHelper(client, []byte(m.config.Pickle), m.config.DBPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
m.cryptoHelper.LoginAs = &mautrix.ReqLogin{
|
||||
Type: mautrix.AuthTypePassword,
|
||||
Identifier: mautrix.UserIdentifier{Type: mautrix.IdentifierTypeUser, User: m.config.UserID},
|
||||
Password: m.config.UserPassword,
|
||||
}
|
||||
if err := m.cryptoHelper.Init(); err != nil {
|
||||
return err
|
||||
}
|
||||
m.client.Crypto = m.cryptoHelper
|
||||
|
||||
m.gptClient = NewGPT(m.config.OpenAIKey)
|
||||
|
||||
m.conversations = make(Conversations, 0)
|
||||
|
||||
m.AddEventHandler(m.InviteHandler())
|
||||
m.AddEventHandler(m.ResponseHandler())
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Bot) Run() error {
|
||||
if err := m.client.Sync(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Bot) Close() error {
|
||||
if err := m.client.Sync(); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := m.cryptoHelper.Close(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Bot) AddEventHandler(eventType event.Type, handler mautrix.EventHandler) {
|
||||
syncer := m.client.Syncer.(*mautrix.DefaultSyncer)
|
||||
syncer.OnEventType(eventType, handler)
|
||||
}
|
||||
|
||||
func (m *Bot) InviteHandler() (event.Type, mautrix.EventHandler) {
|
||||
return event.StateMember, func(source mautrix.EventSource, evt *event.Event) {
|
||||
if evt.GetStateKey() == m.client.UserID.String() && evt.Content.AsMember().Membership == event.MembershipInvite {
|
||||
_, err := m.client.JoinRoomByID(evt.RoomID)
|
||||
if err != nil {
|
||||
m.logger.Error("failed to join room after invite", slog.String("err", err.Error()), slog.String("room_id", evt.RoomID.String()), slog.String("inviter", evt.Sender.String()))
|
||||
return
|
||||
}
|
||||
|
||||
m.logger.Info("Joined room after invite", slog.String("room_id", evt.RoomID.String()), slog.String("inviter", evt.Sender.String()))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (m *Bot) ResponseHandler() (event.Type, mautrix.EventHandler) {
|
||||
return event.EventMessage, func(source mautrix.EventSource, evt *event.Event) {
|
||||
content := evt.Content.AsMessage()
|
||||
eventID := evt.ID
|
||||
m.logger.Info("received message", slog.String("content", content.Body))
|
||||
|
||||
// ignore if the message is already recorded
|
||||
if conv := m.conversations.FindByEventID(eventID); conv != nil {
|
||||
m.logger.Info("known message, ignoring", slog.String("event_id", eventID.String()))
|
||||
return
|
||||
}
|
||||
|
||||
// ignore if the message is sent by the bot itself
|
||||
if evt.Sender == id.UserID(m.config.UserID) {
|
||||
m.logger.Info("message sent by bot itself, ignoring", slog.String("event_id", eventID.String()))
|
||||
return
|
||||
}
|
||||
|
||||
var conv *Conversation
|
||||
// find out if it is a reply to a known conversation
|
||||
parentID := id.EventID("")
|
||||
if relatesTo := content.GetRelatesTo(); relatesTo != nil {
|
||||
if parentID = relatesTo.GetReplyTo(); parentID != "" {
|
||||
m.logger.Info("message is a reply", slog.String("parent_id", parentID.String()))
|
||||
if c := m.conversations.FindByEventID(parentID); c != nil {
|
||||
m.logger.Info("found parent, appending message to conversation", slog.String("event_id", eventID.String()))
|
||||
c.Add(Message{
|
||||
EventID: eventID,
|
||||
ParentID: parentID,
|
||||
Role: openai.ChatMessageRoleUser,
|
||||
Content: content.Body,
|
||||
})
|
||||
conv = c
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// find out if message is addressed to the bot
|
||||
m.logger.Info(content.Body)
|
||||
if conv == nil && strings.HasPrefix(strings.ToLower(content.Body), strings.ToLower(fmt.Sprintf("%s: ", m.config.UserDisplayName))) {
|
||||
m.logger.Info("message is addressed to bot", slog.String("event_id", eventID.String()))
|
||||
conv = NewConversation(eventID, content.Body)
|
||||
m.conversations = append(m.conversations, conv)
|
||||
}
|
||||
|
||||
if conv == nil {
|
||||
m.logger.Info("apparently not for us, ignoring", slog.String("event_id", eventID.String()))
|
||||
return
|
||||
}
|
||||
|
||||
// get reply from GPT
|
||||
reply, err := m.gptClient.Complete(conv)
|
||||
if err != nil {
|
||||
m.logger.Error("failed to get reply from openai", slog.String("err", err.Error()))
|
||||
return
|
||||
}
|
||||
|
||||
formattedReply := format.RenderMarkdown(reply, true, false)
|
||||
formattedReply.RelatesTo = &event.RelatesTo{
|
||||
InReplyTo: &event.InReplyTo{
|
||||
EventID: eventID,
|
||||
},
|
||||
}
|
||||
res, err := m.client.SendMessageEvent(evt.RoomID, event.EventMessage, &formattedReply)
|
||||
if err != nil {
|
||||
m.logger.Error("failed to send message", slog.String("err", err.Error()))
|
||||
return
|
||||
}
|
||||
conv.Add(Message{
|
||||
EventID: res.EventID,
|
||||
ParentID: eventID,
|
||||
Role: openai.ChatMessageRoleAssistant,
|
||||
Content: reply,
|
||||
})
|
||||
|
||||
if len(reply) > 30 {
|
||||
reply = reply[:30] + "..."
|
||||
}
|
||||
m.logger.Info("sent reply", slog.String("parent_id", eventID.String()), slog.String("content", reply))
|
||||
}
|
||||
}
|
|
@ -2,11 +2,20 @@ package bot
|
|||
|
||||
import (
|
||||
"github.com/sashabaranov/go-openai"
|
||||
"maunium.net/go/mautrix"
|
||||
"maunium.net/go/mautrix/id"
|
||||
)
|
||||
|
||||
const systemPrompt = "You are a chatbot that helps people by responding to their questions with short messages."
|
||||
|
||||
type Character struct {
|
||||
UserID string
|
||||
Password string
|
||||
AccessKey string
|
||||
Prompt string
|
||||
client *mautrix.Client
|
||||
}
|
||||
|
||||
type Message struct {
|
||||
EventID id.EventID
|
||||
Role string
|
||||
|
|
173
bot/matrix.go
173
bot/matrix.go
|
@ -1,173 +0,0 @@
|
|||
package bot
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
"github.com/chzyer/readline"
|
||||
"github.com/rs/zerolog"
|
||||
"github.com/sashabaranov/go-openai"
|
||||
"golang.org/x/exp/slog"
|
||||
"maunium.net/go/mautrix"
|
||||
"maunium.net/go/mautrix/crypto/cryptohelper"
|
||||
"maunium.net/go/mautrix/event"
|
||||
"maunium.net/go/mautrix/format"
|
||||
"maunium.net/go/mautrix/id"
|
||||
)
|
||||
|
||||
type Config struct {
|
||||
Homeserver string
|
||||
UserID string
|
||||
UserAccessKey string
|
||||
UserPassword string
|
||||
DBPath string
|
||||
Pickle string
|
||||
OpenAIKey string
|
||||
}
|
||||
|
||||
type Matrix struct {
|
||||
config Config
|
||||
readline *readline.Instance
|
||||
client *mautrix.Client
|
||||
cryptoHelper *cryptohelper.CryptoHelper
|
||||
conversations Conversations
|
||||
gptClient *GPT
|
||||
}
|
||||
|
||||
func New(cfg Config) *Matrix {
|
||||
return &Matrix{
|
||||
config: cfg,
|
||||
}
|
||||
}
|
||||
|
||||
func (m *Matrix) Init() error {
|
||||
client, err := mautrix.NewClient(m.config.Homeserver, id.UserID(m.config.UserID), m.config.UserAccessKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
var oei mautrix.OldEventIgnorer
|
||||
oei.Register(client.Syncer.(mautrix.ExtensibleSyncer))
|
||||
m.client = client
|
||||
|
||||
m.client.Log = zerolog.New(zerolog.NewConsoleWriter(func(w *zerolog.ConsoleWriter) {
|
||||
w.TimeFormat = time.Stamp
|
||||
})).With().Timestamp().Logger().Level(zerolog.InfoLevel)
|
||||
|
||||
m.cryptoHelper, err = cryptohelper.NewCryptoHelper(client, []byte(m.config.Pickle), m.config.DBPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
m.cryptoHelper.LoginAs = &mautrix.ReqLogin{
|
||||
Type: mautrix.AuthTypePassword,
|
||||
Identifier: mautrix.UserIdentifier{Type: mautrix.IdentifierTypeUser, User: m.config.UserID},
|
||||
Password: m.config.UserPassword,
|
||||
}
|
||||
if err := m.cryptoHelper.Init(); err != nil {
|
||||
return err
|
||||
}
|
||||
m.client.Crypto = m.cryptoHelper
|
||||
|
||||
m.gptClient = NewGPT(m.config.OpenAIKey)
|
||||
|
||||
m.conversations = make(Conversations, 0)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Matrix) Run() error {
|
||||
if err := m.client.Sync(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Matrix) Close() error {
|
||||
if err := m.client.Sync(); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := m.cryptoHelper.Close(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (m *Matrix) AddEventHandler(eventType event.Type, handler mautrix.EventHandler) {
|
||||
syncer := m.client.Syncer.(*mautrix.DefaultSyncer)
|
||||
syncer.OnEventType(eventType, handler)
|
||||
}
|
||||
|
||||
func (m *Matrix) InviteHandler(logger *slog.Logger) (event.Type, mautrix.EventHandler) {
|
||||
return event.StateMember, func(source mautrix.EventSource, evt *event.Event) {
|
||||
if evt.GetStateKey() == m.client.UserID.String() && evt.Content.AsMember().Membership == event.MembershipInvite {
|
||||
_, err := m.client.JoinRoomByID(evt.RoomID)
|
||||
if err != nil {
|
||||
logger.Error("failed to join room after invite", slog.String("err", err.Error()), slog.String("room_id", evt.RoomID.String()), slog.String("inviter", evt.Sender.String()))
|
||||
return
|
||||
}
|
||||
|
||||
logger.Info("Joined room after invite", slog.String("room_id", evt.RoomID.String()), slog.String("inviter", evt.Sender.String()))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (m *Matrix) ResponseHandler(logger *slog.Logger) (event.Type, mautrix.EventHandler) {
|
||||
return event.EventMessage, func(source mautrix.EventSource, evt *event.Event) {
|
||||
content := evt.Content.AsMessage()
|
||||
eventID := evt.ID
|
||||
logger.Info("received message", slog.String("content", content.Body))
|
||||
|
||||
conv := m.conversations.FindByEventID(eventID)
|
||||
if conv != nil {
|
||||
logger.Info("known message, ignoring", slog.String("event_id", eventID.String()))
|
||||
return
|
||||
}
|
||||
|
||||
parentID := id.EventID("")
|
||||
if relatesTo := content.GetRelatesTo(); relatesTo != nil {
|
||||
parentID = relatesTo.GetReplyTo()
|
||||
}
|
||||
if parentID != "" {
|
||||
logger.Info("parent found, looking for conversation", slog.String("parent_id", parentID.String()))
|
||||
conv = m.conversations.FindByEventID(parentID)
|
||||
}
|
||||
if conv != nil {
|
||||
conv.Add(Message{
|
||||
EventID: eventID,
|
||||
ParentID: parentID,
|
||||
Role: openai.ChatMessageRoleUser,
|
||||
Content: content.Body,
|
||||
})
|
||||
logger.Info("found parent, appending message to conversation", slog.String("event_id", eventID.String()))
|
||||
} else {
|
||||
conv = NewConversation(eventID, content.Body)
|
||||
m.conversations = append(m.conversations, conv)
|
||||
logger.Info("no parent found, starting new conversation", slog.String("event_id", eventID.String()))
|
||||
}
|
||||
|
||||
if evt.Sender != id.UserID(m.config.UserID) {
|
||||
// get reply from GPT
|
||||
reply, err := m.gptClient.Complete(conv)
|
||||
if err != nil {
|
||||
logger.Error("failed to get reply from openai", slog.String("err", err.Error()))
|
||||
return
|
||||
}
|
||||
|
||||
formattedReply := format.RenderMarkdown(reply, true, false)
|
||||
formattedReply.RelatesTo = &event.RelatesTo{
|
||||
InReplyTo: &event.InReplyTo{
|
||||
EventID: eventID,
|
||||
},
|
||||
}
|
||||
if _, err := m.client.SendMessageEvent(evt.RoomID, event.EventMessage, &formattedReply); err != nil {
|
||||
logger.Error("failed to send message", slog.String("err", err.Error()))
|
||||
return
|
||||
}
|
||||
|
||||
if len(reply) > 30 {
|
||||
reply = reply[:30] + "..."
|
||||
}
|
||||
logger.Info("sent reply", slog.String("parent_id", eventID.String()), slog.String("content", reply))
|
||||
}
|
||||
}
|
||||
}
|
20
main.go
20
main.go
|
@ -14,14 +14,15 @@ func main() {
|
|||
logger := slog.New(slog.NewTextHandler(os.Stderr, nil))
|
||||
|
||||
matrixClient := bot.New(bot.Config{
|
||||
Homeserver: getParam("MATRIX_HOMESERVER", "http://localhost"),
|
||||
UserID: getParam("MATRIX_USER_ID", "@bot:localhost"),
|
||||
UserPassword: getParam("MATRIX_PASSWORD", "secret"),
|
||||
UserAccessKey: getParam("MATRIX_ACCESS_KEY", "secret"),
|
||||
DBPath: getParam("BOT_DB_PATH", "bot.db"),
|
||||
Pickle: getParam("BOT_PICKLE", "scrambled"),
|
||||
OpenAIKey: getParam("OPENAI_API_KEY", "no key"),
|
||||
})
|
||||
Homeserver: getParam("MATRIX_HOMESERVER", "http://localhost"),
|
||||
UserID: getParam("MATRIX_USER_ID", "@bot:localhost"),
|
||||
UserPassword: getParam("MATRIX_PASSWORD", "secret"),
|
||||
UserAccessKey: getParam("MATRIX_ACCESS_KEY", "secret"),
|
||||
UserDisplayName: getParam("MATRIX_DISPLAY_NAME", "Bot"),
|
||||
DBPath: getParam("BOT_DB_PATH", "bot.db"),
|
||||
Pickle: getParam("BOT_PICKLE", "scrambled"),
|
||||
OpenAIKey: getParam("OPENAI_API_KEY", "no key"),
|
||||
}, logger)
|
||||
|
||||
if err := matrixClient.Init(); err != nil {
|
||||
logger.Error(err.Error())
|
||||
|
@ -29,9 +30,6 @@ func main() {
|
|||
}
|
||||
go matrixClient.Run()
|
||||
|
||||
matrixClient.AddEventHandler(matrixClient.InviteHandler(logger))
|
||||
matrixClient.AddEventHandler(matrixClient.ResponseHandler(logger))
|
||||
|
||||
done := make(chan os.Signal)
|
||||
signal.Notify(done, os.Interrupt)
|
||||
<-done
|
||||
|
|
Loading…
Reference in New Issue