// 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'", 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 { 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, append([]string{os.Args[0]}, args...)) } else { fmt.Fprintf(os.Stderr, "Unknown command: %s\n", args[0]) } //cmd.Run(ctx, append([]string{os.Args[0]}, args...)) } fmt.Println("Stdin closed.") return nil }, Commands: []*cli.Command{ { Name: "message", Usage: "send a message using ", 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
", 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() }