183 lines
5.2 KiB
Go
183 lines
5.2 KiB
Go
// from https://github.com/tulir/whatsmeow/issues/659
|
|
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/google/shlex"
|
|
_ "github.com/mattn/go-sqlite3"
|
|
"github.com/mdp/qrterminal/v3"
|
|
"github.com/urfave/cli/v3"
|
|
"go.mau.fi/whatsmeow"
|
|
"go.mau.fi/whatsmeow/proto/waE2E"
|
|
"go.mau.fi/whatsmeow/store/sqlstore"
|
|
"go.mau.fi/whatsmeow/types"
|
|
"go.mau.fi/whatsmeow/types/events"
|
|
waLog "go.mau.fi/whatsmeow/util/log"
|
|
"google.golang.org/protobuf/proto"
|
|
)
|
|
|
|
func eventHandler(evt interface{}) {
|
|
switch v := evt.(type) {
|
|
case *events.Message:
|
|
fmt.Println("Received a message from !", v.Info.Sender)
|
|
|
|
fmt.Println("Received a message!", v.Message.GetConversation())
|
|
}
|
|
}
|
|
|
|
func initClient() (*whatsmeow.Client, *sqlstore.Container, waLog.Logger) {
|
|
dbLog := waLog.Stdout("Database", "DEBUG", true)
|
|
// Make sure you add appropriate DB connector imports, e.g. github.com/mattn/go-sqlite3 for SQLite
|
|
container, err := sqlstore.New(context.Background(), "sqlite3", "file:sqlite3.db?_foreign_keys=on", dbLog)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// If you want multiple sessions, remember their JIDs and use .GetDevice(jid) or .GetAllDevices() instead.
|
|
deviceStore, err := container.GetFirstDevice(context.Background())
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
clientLog := waLog.Stdout("Client", "DEBUG", true)
|
|
client := whatsmeow.NewClient(deviceStore, clientLog)
|
|
client.AddEventHandler(eventHandler)
|
|
|
|
// handle login via QR
|
|
if client.Store.ID == nil {
|
|
fmt.Println("Client store ID is nil, scanning QR")
|
|
// No ID stored, new login
|
|
qrChan, _ := client.GetQRChannel(context.Background())
|
|
err = client.Connect()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
for evt := range qrChan {
|
|
if evt.Event == "code" {
|
|
// Render the QR code here
|
|
// e.g. qrterminal.GenerateHalfBlock(evt.Code, qrterminal.L, os.Stdout)
|
|
qrterminal.GenerateHalfBlock(evt.Code, qrterminal.L, os.Stdout)
|
|
// or just manually `echo 2@... | qrencode -t ansiutf8` in a terminal
|
|
fmt.Println("QR code:", evt.Code)
|
|
} else {
|
|
fmt.Println("Login event:", evt.Event)
|
|
}
|
|
}
|
|
} else {
|
|
// Already logged in, just connect
|
|
fmt.Println("Connecting")
|
|
|
|
err = client.Connect()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
return client, container, dbLog
|
|
}
|
|
|
|
func sendMessage(message *waE2E.Message, jidStr string, client *whatsmeow.Client) {
|
|
if message != nil {
|
|
var JID types.JID
|
|
JID, err := types.ParseJID(jidStr)
|
|
if err != nil {
|
|
log.Fatal("Failed to parse JID: ", jidStr)
|
|
} else {
|
|
log.Println("Parsed JID correctly", JID)
|
|
}
|
|
_, err = client.SendMessage(context.Background(), JID, message)
|
|
// FIXME showing error even when successful
|
|
if err == nil {
|
|
log.Println("Sent Message successfully")
|
|
} else {
|
|
fmt.Println("Failed to Send Message", err)
|
|
|
|
}
|
|
}
|
|
}
|
|
func main() {
|
|
var jidStr, header, text string
|
|
var message *waE2E.Message
|
|
var client *whatsmeow.Client
|
|
var container *sqlstore.Container
|
|
//var dbLog waLog.Logger
|
|
cmd := &cli.Command{
|
|
Usage: "Run WhatsApp actions from your CLI. User JID has to end with '@s.whatsapp.net', Group ID with '@g.us'." +
|
|
"Defaults to listening on stdin for batch processing.",
|
|
Action: func(ctx context.Context, cmd *cli.Command) error {
|
|
fmt.Println("No command specified. Reading from stdin. Press Ctrl+D to exit or run with --help to get help.")
|
|
scanner := bufio.NewScanner(os.Stdin)
|
|
for scanner.Scan() {
|
|
if err := scanner.Err(); err != nil {
|
|
fmt.Fprintf(os.Stderr, "Error reading: %v\n", err)
|
|
continue
|
|
}
|
|
args, err := shlex.Split(scanner.Text())
|
|
if err != nil || len(args) == 0 {
|
|
fmt.Fprintf(os.Stderr, "Error parsing command: %v\n", err)
|
|
continue
|
|
}
|
|
fmt.Println(args)
|
|
subcmd := cmd.Command(args[0])
|
|
if subcmd != nil {
|
|
subcmd.Run(ctx, args)
|
|
} else {
|
|
fmt.Fprintf(os.Stderr, "Unknown command: %s\n", args[0])
|
|
}
|
|
}
|
|
fmt.Println("Stdin closed.")
|
|
return nil
|
|
},
|
|
Commands: []*cli.Command{
|
|
{
|
|
Name: "message",
|
|
Usage: "send a message using <JID> <MESSAGE>",
|
|
Arguments: []cli.Argument{
|
|
&cli.StringArg{Name: "jid", Destination: &jidStr},
|
|
&cli.StringArg{Name: "message", Destination: &text},
|
|
},
|
|
Action: func(ctx context.Context, cmd *cli.Command) error {
|
|
message = &waE2E.Message{Conversation: proto.String(text)}
|
|
sendMessage(message, jidStr, client)
|
|
return nil
|
|
},
|
|
},
|
|
{
|
|
Name: "getgroups",
|
|
Usage: "print all available group info",
|
|
Action: func(ctx context.Context, cmd *cli.Command) error {
|
|
fmt.Println(client.GetJoinedGroups(ctx))
|
|
return nil
|
|
},
|
|
},
|
|
{
|
|
Name: "poll",
|
|
Usage: "send a poll using <JID> <HEADER> <OPTIONS>",
|
|
Arguments: []cli.Argument{
|
|
&cli.StringArg{Name: "jid", Destination: &jidStr}, // use id field of group
|
|
&cli.StringArg{Name: "header", Destination: &header},
|
|
&cli.StringArgs{Name: "options", Min: 2, Max: -1},
|
|
},
|
|
Action: func(ctx context.Context, cmd *cli.Command) error {
|
|
// parse JID
|
|
message = client.BuildPollCreation(header, cmd.StringArgs("options"), 1)
|
|
sendMessage(message, jidStr, client)
|
|
return nil
|
|
},
|
|
},
|
|
},
|
|
}
|
|
// before
|
|
client, container, _ = initClient()
|
|
// run
|
|
cmd.Run(context.Background(), os.Args)
|
|
// after
|
|
time.Sleep(5 * time.Second)
|
|
client.Disconnect()
|
|
container.Close()
|
|
}
|