Files
LocalAGI/services/connectors/twitter.go
Ettore Di Giacinto f0bd184fbd feat: add twitter action and connector (#58)
* feat: add twitter post action

Signed-off-by: mudler <mudler@localai.io>

* feat: handle twitter post messages limits

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>

* feat: add twitter connector, unify client

Signed-off-by: Ettore Di Giacinto <mudler@localai.io>

* make sure answers do not exceed twitter maximum

---------

Signed-off-by: mudler <mudler@localai.io>
Signed-off-by: Ettore Di Giacinto <mudler@localai.io>
2025-03-19 22:14:32 +01:00

136 lines
2.8 KiB
Go

package connectors
import (
"context"
"fmt"
"os"
"os/signal"
"github.com/mudler/LocalAgent/core/agent"
"github.com/mudler/LocalAgent/pkg/xlog"
"github.com/mudler/LocalAgent/services/connectors/twitter"
"github.com/sashabaranov/go-openai"
)
type Twitter struct {
token string
botUsername string
client *twitter.TwitterClient
noCharacterLimit bool
}
func (t *Twitter) AgentResultCallback() func(state agent.ActionState) {
return func(state agent.ActionState) {
}
}
func (t *Twitter) AgentReasoningCallback() func(state agent.ActionCurrentState) bool {
return func(state agent.ActionCurrentState) bool {
return true
}
}
func NewTwitterConnector(config map[string]string) (*Twitter, error) {
return &Twitter{
token: config["token"],
botUsername: config["botUsername"],
client: twitter.NewTwitterClient(config["token"]),
noCharacterLimit: config["noCharacterLimit"] == "true",
}, nil
}
func (t *Twitter) Start(a *agent.Agent) {
ctx, cancel := signal.NotifyContext(a.Context(), os.Interrupt)
defer cancel()
// Step 1: Setup stream rules
xlog.Info("Setting up stream rules...")
err := t.client.AddStreamRule(t.botUsername)
if err != nil {
xlog.Error("Failed to add stream rule:", err)
}
// Step 2: Listen for mentions and respond
fmt.Println("Listening for mentions...")
go t.loop(ctx, a)
}
func (t *Twitter) loop(ctx context.Context, a *agent.Agent) {
for {
select {
case <-ctx.Done():
xlog.Info("Shutting down Twitter connector...")
return
default:
if err := t.run(a); err != nil {
xlog.Error("Error running Twitter connector", "err", err)
return
}
}
}
}
func (t *Twitter) run(a *agent.Agent) error {
tweet, err := t.client.ListenForMentions()
if err != nil {
xlog.Error("Error getting mention", "error", err)
return nil
}
xlog.Info("Got mention", "tweet", tweet)
// Check if bot has already replied
hasReplied, err := t.client.HasReplied(tweet.ID, t.botUsername)
if err != nil {
xlog.Error("Error checking if bot has replied", "error", err)
return nil
}
if hasReplied {
xlog.Info("Bot has already replied to this tweet")
return nil
}
res := a.Ask(
agent.WithConversationHistory(
[]openai.ChatCompletionMessage{
{
Role: "system",
Content: "You are replying to a twitter mention, keep answer short",
},
{
Role: "user",
Content: tweet.Text,
},
},
),
)
if res.Error != nil {
xlog.Error("Error getting response from agent", "error", res.Error)
return nil
}
if len(res.Response) > 280 && !t.noCharacterLimit {
xlog.Error("Tweet is too long, max 280 characters")
return nil
}
// Reply to tweet
err = t.client.ReplyToTweet(tweet.ID, res.Response)
if err != nil {
xlog.Error("Error replying to tweet", "error", err)
return nil
}
xlog.Debug("Replied successfully!")
return nil
}