package main import ( "fmt" "log" "os" "strings" "github.com/slack-go/slack/socketmode" "github.com/slack-go/slack" "github.com/slack-go/slack/slackevents" ) func main() { appToken := os.Getenv("SLACK_APP_TOKEN") if appToken == "" { fmt.Fprintf(os.Stderr, "SLACK_APP_TOKEN must be set.\n") os.Exit(1) } if !strings.HasPrefix(appToken, "xapp-") { fmt.Fprintf(os.Stderr, "SLACK_APP_TOKEN must have the prefix \"xapp-\".") } botToken := os.Getenv("SLACK_BOT_TOKEN") if botToken == "" { fmt.Fprintf(os.Stderr, "SLACK_BOT_TOKEN must be set.\n") os.Exit(1) } if !strings.HasPrefix(botToken, "xoxb-") { fmt.Fprintf(os.Stderr, "SLACK_BOT_TOKEN must have the prefix \"xoxb-\".") } api := slack.New( botToken, slack.OptionDebug(true), slack.OptionLog(log.New(os.Stdout, "api: ", log.Lshortfile|log.LstdFlags)), slack.OptionAppLevelToken(appToken), ) client := socketmode.New( api, socketmode.OptionDebug(true), socketmode.OptionLog(log.New(os.Stdout, "socketmode: ", log.Lshortfile|log.LstdFlags)), ) go func() { for evt := range client.Events { switch evt.Type { case socketmode.EventTypeConnecting: fmt.Println("Connecting to Slack with Socket Mode...") case socketmode.EventTypeConnectionError: fmt.Println("Connection failed. Retrying later...") case socketmode.EventTypeConnected: fmt.Println("Connected to Slack with Socket Mode.") case socketmode.EventTypeEventsAPI: eventsAPIEvent, ok := evt.Data.(slackevents.EventsAPIEvent) if !ok { fmt.Printf("Ignored %+v\n", evt) continue } fmt.Printf("Event received: %+v\n", eventsAPIEvent) client.Ack(*evt.Request) switch eventsAPIEvent.Type { case slackevents.CallbackEvent: innerEvent := eventsAPIEvent.InnerEvent switch ev := innerEvent.Data.(type) { case *slackevents.AppMentionEvent: _, _, err := client.PostMessage(ev.Channel, slack.MsgOptionText("Yes, hello.", false)) if err != nil { fmt.Printf("failed posting message: %v", err) } case *slackevents.MemberJoinedChannelEvent: fmt.Printf("user %q joined to channel %q", ev.User, ev.Channel) } default: client.Debugf("unsupported Events API event received") } case socketmode.EventTypeInteractive: callback, ok := evt.Data.(slack.InteractionCallback) if !ok { fmt.Printf("Ignored %+v\n", evt) continue } fmt.Printf("Interaction received: %+v\n", callback) var payload interface{} switch callback.Type { case slack.InteractionTypeBlockActions: // See https://api.slack.com/apis/connections/socket-implement#button client.Debugf("button clicked!") case slack.InteractionTypeShortcut: case slack.InteractionTypeViewSubmission: // See https://api.slack.com/apis/connections/socket-implement#modal case slack.InteractionTypeDialogSubmission: default: } client.Ack(*evt.Request, payload) case socketmode.EventTypeSlashCommand: cmd, ok := evt.Data.(slack.SlashCommand) if !ok { fmt.Printf("Ignored %+v\n", evt) continue } client.Debugf("Slash command received: %+v", cmd) payload := map[string]interface{}{ "blocks": []slack.Block{ slack.NewSectionBlock( &slack.TextBlockObject{ Type: slack.MarkdownType, Text: "foo", }, nil, slack.NewAccessory( slack.NewButtonBlockElement( "", "somevalue", &slack.TextBlockObject{ Type: slack.PlainTextType, Text: "bar", }, ), ), ), }, } client.Ack(*evt.Request, payload) default: fmt.Fprintf(os.Stderr, "Unexpected event type received: %s\n", evt.Type) } } }() client.Run() }