telegabber/telegram/commands.go

1232 lines
33 KiB
Go
Raw Permalink Normal View History

2019-11-24 17:10:29 +00:00
package telegram
2019-11-26 00:04:11 +00:00
import (
2019-12-05 19:56:12 +00:00
"fmt"
2019-12-05 18:13:17 +00:00
"github.com/pkg/errors"
"sort"
2019-12-04 23:10:08 +00:00
"strconv"
2019-11-26 00:04:11 +00:00
"strings"
2019-12-08 14:24:51 +00:00
"time"
"unicode"
2019-12-04 23:10:08 +00:00
"dev.narayana.im/narayana/telegabber/xmpp/gateway"
2019-12-05 23:21:39 +00:00
log "github.com/sirupsen/logrus"
2022-01-17 20:45:40 +00:00
"github.com/zelenin/go-tdlib/client"
2019-11-26 00:04:11 +00:00
)
2024-02-10 18:46:02 +00:00
const unknownCommand string = "Unknown command"
2019-11-24 17:10:29 +00:00
const notEnoughArguments string = "Not enough arguments"
2023-08-28 14:16:57 +00:00
const TelegramNotInitialized string = "Telegram connection is not initialized yet"
const TelegramAuthDone string = "Authorization is done already"
const notOnline string = "Not online"
2022-02-08 20:25:58 +00:00
var permissionsAdmin = client.ChatAdministratorRights{
2022-02-03 18:51:27 +00:00
CanChangeInfo: true,
CanPostMessages: true,
CanEditMessages: true,
CanDeleteMessages: true,
CanInviteUsers: true,
CanRestrictMembers: true,
CanPinMessages: true,
CanPromoteMembers: false,
}
2022-02-01 04:57:17 +00:00
var permissionsMember = client.ChatPermissions{
CanSendBasicMessages: true,
CanSendAudios: true,
CanSendDocuments: true,
CanSendPhotos: true,
CanSendVideos: true,
CanSendVideoNotes: true,
CanSendVoiceNotes: true,
2022-02-01 04:57:17 +00:00
CanSendPolls: true,
CanSendOtherMessages: true,
CanAddWebPagePreviews: true,
CanChangeInfo: true,
CanInviteUsers: true,
CanPinMessages: true,
CanManageTopics: true,
2022-02-01 04:57:17 +00:00
}
var permissionsReadonly = client.ChatPermissions{}
2019-11-24 17:10:29 +00:00
2019-11-26 00:04:11 +00:00
var transportCommands = map[string]command{
2024-02-15 09:40:57 +00:00
"help": command{0, []string{}, "help", nil},
"login": command{1, []string{"phone"}, "sign in", nil},
"logout": command{0, []string{}, "sign out", nil},
"cancelauth": command{0, []string{}, "quit the signin wizard", nil},
"code": command{1, []string{"xxxxx"}, "check one-time code", nil},
"password": command{1, []string{"********"}, "check 2fa password", nil},
"setusername": command{0, []string{"@username"}, "update @username", nil},
"setname": command{1, []string{"first", "last"}, "update name", nil},
"setbio": command{0, []string{"Lorem ipsum"}, "update about", nil},
"setpassword": command{0, []string{"old", "new"}, "set or remove password", nil},
"config": command{0, []string{"param", "value"}, "view or update configuration options", nil},
"report": command{2, []string{"chat", "comment"}, "report a chat by id or @username", nil},
"add": command{1, []string{"@username"}, "add @username to your chat list", nil},
"join": command{1, []string{"https://t.me/invite_link"}, "join to chat via invite link or @publicname", nil},
"supergroup": command{1, []string{"title", "description"}, "create new supergroup «title» with «description»", nil},
"channel": command{1, []string{"title", "description"}, "create new channel «title» with «description»", nil},
2019-11-26 00:04:11 +00:00
}
2024-02-15 09:40:57 +00:00
var notForGroups = []ChatType{ChatTypeBasicGroup, ChatTypeSupergroup, ChatTypeChannel}
var notForPM = []ChatType{ChatTypePrivate, ChatTypeSecret}
var notForPMAndBasic = []ChatType{ChatTypePrivate, ChatTypeSecret, ChatTypeBasicGroup}
2024-02-15 09:40:57 +00:00
var onlyForSecret = []ChatType{ChatTypePrivate, ChatTypeBasicGroup, ChatTypeSupergroup, ChatTypeChannel}
2019-11-26 00:04:11 +00:00
var chatCommands = map[string]command{
2024-02-15 09:40:57 +00:00
"help": command{0, []string{}, "help", nil},
"d": command{0, []string{"n"}, "delete your last message(s)", nil},
"s": command{1, []string{"edited message"}, "edit your last message", nil},
"silent": command{1, []string{"message"}, "send a message without sound", nil},
"schedule": command{2, []string{"{online | 2006-01-02T15:04:05 | 15:04:05}", "message"}, "schedules a message either to timestamp or to whenever the user goes online", nil},
"forward": command{2, []string{"message_id", "target_chat"}, "forwards a message", nil},
"vcard": command{0, []string{}, "print vCard as text", nil},
"add": command{1, []string{"@username"}, "add @username to your chat list", nil},
"join": command{1, []string{"https://t.me/invite_link"}, "join to chat via invite link or @publicname", nil},
"group": command{1, []string{"title"}, "create groupchat «title» with current user", &notForGroups},
"supergroup": command{1, []string{"title", "description"}, "create new supergroup «title» with «description»", nil},
"channel": command{1, []string{"title", "description"}, "create new channel «title» with «description»", nil},
"secret": command{0, []string{}, "create secretchat with current user", &notForGroups},
"search": command{0, []string{"string", "[limit]"}, "search <string> in current chat", nil},
"history": command{0, []string{"limit"}, "get last [limit] messages from current chat", nil},
"block": command{0, []string{}, "blacklist current user", &notForGroups},
"unblock": command{0, []string{}, "unblacklist current user", &notForGroups},
"invite": command{1, []string{"id or @username"}, "add user to current chat", &notForPM},
"link": command{0, []string{}, "get invite link for current chat", &notForPM},
"kick": command{1, []string{"id or @username"}, "remove user from current chat", &notForPM},
2024-05-09 23:32:57 +00:00
"mute": command{0, []string{"id or @username", "hours"}, "mute the whole chat or a user in current chat", &notForPMAndBasic},
"unmute": command{0, []string{"id or @username"}, "unmute the whole chat or a user in the current chat", &notForPMAndBasic},
2024-02-15 09:40:57 +00:00
"ban": command{1, []string{"id or @username", "hours"}, "restrict @username from current chat for [hours] or forever", &notForPM},
"unban": command{1, []string{"id or @username"}, "unbans @username in current chat (and devotes from admins)", &notForPM},
"promote": command{1, []string{"id or @username", "title"}, "promote user to admin in current chat", &notForPM},
"leave": command{0, []string{}, "leave current chat", &notForPM},
"leave!": command{0, []string{}, "leave current chat (for owners)", &notForPM},
"ttl": command{0, []string{"seconds"}, "set secret chat messages TTL before self-destroying", &onlyForSecret},
"close": command{0, []string{}, "close current secret chat", &onlyForSecret},
"delete": command{0, []string{}, "delete current chat from chat list", nil},
"members": command{0, []string{"query"}, "search members [by optional query] in current chat (requires admin rights)", nil},
2019-11-26 00:04:11 +00:00
}
var transportConfigurationOptions = map[string]configurationOption{
2022-02-08 20:25:58 +00:00
"timezone": configurationOption{"<timezone>", "adjust timezone for Telegram user statuses (example: +02:00)"},
"keeponline": configurationOption{"<bool>", "always keep telegram session online and rely on jabber offline messages (example: true)"},
2022-02-08 18:49:49 +00:00
"rawmessages": configurationOption{"<bool>", "do not add additional info (message id, origin etc.) to incoming messages (example: true)"},
2019-11-26 00:04:11 +00:00
}
type command struct {
RequiredArgs int
Arguments []string
Description string
2024-02-15 09:40:57 +00:00
NotFor *[]ChatType
}
type configurationOption struct {
2019-11-26 00:04:11 +00:00
arguments string
description string
}
// CommandType disinguishes command sets by chat
type CommandType int
2019-11-26 00:04:11 +00:00
const (
CommandTypeTransport CommandType = iota
CommandTypeChat
2019-11-26 00:04:11 +00:00
)
// GetCommands exposes the set of commands
func GetCommands(typ CommandType) map[string]command {
2019-11-26 00:04:11 +00:00
var commandMap map[string]command
switch typ {
case CommandTypeTransport:
2019-11-26 00:04:11 +00:00
commandMap = transportCommands
case CommandTypeChat:
2019-11-26 00:04:11 +00:00
commandMap = chatCommands
}
return commandMap
}
// GetCommand obtains one command
func GetCommand(typ CommandType, cmd string) (command, bool) {
commands := GetCommands(typ)
command, ok := commands[cmd]
return command, ok
}
// SortedCommandKeys sorts a slice with command keys
func SortedCommandKeys(commandMap map[string]command) []string {
keys := make([]string, len(commandMap))
i := 0
for k := range commandMap {
keys[i] = k
i++
}
sort.Strings(keys)
return keys
}
// CommandToHelpString builds a text description of a command
func CommandToHelpString(name string, cmd command) string {
var str strings.Builder
str.WriteString("/")
str.WriteString(name)
for i, arg := range cmd.Arguments {
optional := i >= cmd.RequiredArgs
str.WriteString(" ")
if optional {
str.WriteString("[")
}
str.WriteString(arg)
if optional {
str.WriteString("]")
}
}
str.WriteString(" — ")
str.WriteString(cmd.Description)
return str.String()
}
2024-02-15 09:40:57 +00:00
// IsCommandFor checks the suitability of a command for a chat type
func IsCommandForChatType(cmd command, chatType ChatType) bool {
if cmd.NotFor != nil {
for _, typ := range *cmd.NotFor {
if chatType == typ {
return false
}
}
}
return true
}
func (c *Client) helpString(typ CommandType, chatId int64) string {
var str strings.Builder
commandMap := GetCommands(typ)
2024-02-15 09:40:57 +00:00
chatType, chatTypeErr := c.GetChatType(chatId)
2019-11-26 00:04:11 +00:00
str.WriteString("Available commands:\n")
for _, name := range SortedCommandKeys(commandMap) {
command := commandMap[name]
2024-02-15 09:40:57 +00:00
if chatTypeErr == nil && !IsCommandForChatType(command, chatType) {
continue
}
str.WriteString(CommandToHelpString(name, command))
2019-11-26 00:04:11 +00:00
str.WriteString("\n")
}
if typ == CommandTypeTransport {
2019-11-26 00:04:11 +00:00
str.WriteString("Configuration options\n")
for name, option := range transportConfigurationOptions {
str.WriteString(name)
str.WriteString(" ")
str.WriteString(option.arguments)
str.WriteString(" — ")
str.WriteString(option.description)
str.WriteString("\n")
}
}
str.WriteString("\nYou may use ! instead of / if it conflicts with internal commands of a client")
2019-11-26 00:04:11 +00:00
return str.String()
}
2019-12-03 00:32:53 +00:00
func parseCommand(cmdline string) (string, []string) {
bodyFields := strings.Fields(cmdline)
return bodyFields[0][1:], bodyFields[1:]
}
2022-02-08 20:25:58 +00:00
func rawCmdArguments(cmdline string, start uint8) string {
var state uint
// /cmd ababa galamaga
// 01 2 3 45
2022-02-08 20:25:58 +00:00
startState := uint(3 + 2*start)
for i, r := range cmdline {
2022-02-08 20:25:58 +00:00
isOdd := state%2 == 1
isSpace := unicode.IsSpace(r)
if (!isOdd && !isSpace) || (isOdd && isSpace) {
state += 1
}
if state == startState {
return cmdline[i:]
}
}
return ""
}
2023-06-01 20:37:38 +00:00
func keyValueString(key, value string) string {
return fmt.Sprintf("%s: %s", key, value)
}
func (c *Client) unsubscribe(chatID int64) error {
2024-01-24 23:52:40 +00:00
args := gateway.SimplePresence(chatID, "unsubscribed")
return c.sendPresence(args...)
2019-12-08 15:35:27 +00:00
}
2019-12-08 16:19:35 +00:00
func (c *Client) sendMessagesReverse(chatID int64, messages []*client.Message) {
for i := len(messages) - 1; i >= 0; i-- {
message := messages[i]
reply, _ := c.getMessageReply(message, false, true)
2019-12-08 16:19:35 +00:00
gateway.SendMessage(
c.jid,
strconv.FormatInt(chatID, 10),
c.formatMessage(0, 0, false, message),
strconv.FormatInt(message.Id, 10),
2019-12-08 16:19:35 +00:00
c.xmpp,
2023-03-05 08:00:53 +00:00
reply,
"",
2023-03-18 21:43:11 +00:00
false,
false,
2019-12-08 16:19:35 +00:00
)
}
}
2021-12-04 18:10:54 +00:00
func (c *Client) usernameOrIDToID(username string) (int64, error) {
userID, err := strconv.ParseInt(username, 10, 64)
2019-12-08 13:58:17 +00:00
// couldn't parse the id, try to lookup as a username
if err != nil {
chat, err := c.client.SearchPublicChat(&client.SearchPublicChatRequest{
Username: username,
})
if err != nil {
return 0, err
}
2022-01-17 20:45:40 +00:00
userID = chat.Id
2021-12-04 18:10:54 +00:00
if userID <= 0 {
2019-12-08 13:58:17 +00:00
return 0, errors.New("Not a user")
}
}
2021-12-04 18:10:54 +00:00
return userID, nil
2019-12-08 13:58:17 +00:00
}
2019-12-03 00:32:53 +00:00
// ProcessTransportCommand executes a command sent directly to the component
2024-02-18 09:36:23 +00:00
// and returns a response and execution success result
func (c *Client) ProcessTransportCommand(cmdline string, resource string) (string, bool) {
2019-12-03 00:32:53 +00:00
cmd, args := parseCommand(cmdline)
command, ok := transportCommands[cmd]
if !ok {
2024-02-18 09:36:23 +00:00
return unknownCommand, false
}
if len(args) < command.RequiredArgs {
2024-02-18 09:36:23 +00:00
return notEnoughArguments, false
}
2019-11-24 17:10:29 +00:00
switch cmd {
case "login", "code", "password":
if cmd == "login" && c.Session.Login != "" {
2024-02-18 09:36:23 +00:00
return "Phone number already provided, use /cancelauth to start over", false
2019-11-24 17:10:29 +00:00
}
if cmd == "login" {
2023-08-28 14:16:57 +00:00
err := c.TryLogin(resource, args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
2023-08-31 22:24:30 +00:00
c.locks.authorizerWriteLock.Lock()
defer c.locks.authorizerWriteLock.Unlock()
2023-08-28 14:16:57 +00:00
c.authorizer.PhoneNumber <- args[0]
} else {
2023-08-31 22:24:30 +00:00
c.locks.authorizerWriteLock.Lock()
defer c.locks.authorizerWriteLock.Unlock()
2023-08-28 14:16:57 +00:00
if c.authorizer == nil {
2024-02-18 09:36:23 +00:00
return TelegramNotInitialized, false
2023-08-28 14:16:57 +00:00
}
2019-11-24 17:10:29 +00:00
2023-08-28 14:16:57 +00:00
if c.authorizer.isClosed {
2024-02-18 09:36:23 +00:00
return TelegramAuthDone, false
2023-08-28 14:16:57 +00:00
}
2023-08-28 14:16:57 +00:00
switch cmd {
// check auth code
case "code":
c.authorizer.Code <- args[0]
// check auth password
case "password":
c.authorizer.Password <- args[0]
}
2019-11-24 17:10:29 +00:00
}
2019-12-04 23:10:08 +00:00
// sign out
case "logout":
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, false
}
2019-12-04 23:10:08 +00:00
_, err := c.client.LogOut()
if err != nil {
2024-02-18 09:36:23 +00:00
return errors.Wrap(err, "Logout error").Error(), false
2019-12-04 23:10:08 +00:00
}
for _, id := range c.cache.ChatsKeys() {
c.unsubscribe(id)
}
2019-12-04 23:10:08 +00:00
c.Session.Login = ""
2023-06-16 04:34:49 +00:00
// cancel auth
case "cancelauth":
if c.Online() {
2024-02-18 09:36:23 +00:00
return "Not allowed when online, use /logout instead", false
2023-06-16 04:34:49 +00:00
}
c.cancelAuth()
2024-02-18 09:36:23 +00:00
return "Cancelled", true
2019-12-05 18:13:17 +00:00
// set @username
case "setusername":
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, false
}
2019-12-05 18:13:17 +00:00
var username string
if len(args) > 0 {
username = args[0]
}
_, err := c.client.SetUsername(&client.SetUsernameRequest{
Username: username,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return errors.Wrap(err, "Couldn't set username").Error(), false
2019-12-05 18:13:17 +00:00
}
// set My Name
case "setname":
firstname := args[0]
2019-12-05 18:13:17 +00:00
var lastname string
2022-08-15 10:10:29 +00:00
if firstname == "" {
2024-02-18 09:36:23 +00:00
return "The name should contain at least one character", false
2022-08-15 10:10:29 +00:00
}
2019-12-05 18:13:17 +00:00
if len(args) > 1 {
2022-08-15 10:10:29 +00:00
lastname = rawCmdArguments(cmdline, 1)
2019-12-05 18:13:17 +00:00
}
c.locks.authorizerWriteLock.Lock()
2022-08-15 10:10:29 +00:00
if c.authorizer != nil && !c.authorizer.isClosed {
c.authorizer.FirstName <- firstname
c.authorizer.LastName <- lastname
c.locks.authorizerWriteLock.Unlock()
2022-08-15 10:10:29 +00:00
} else {
c.locks.authorizerWriteLock.Unlock()
2022-08-15 10:10:29 +00:00
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, false
2022-08-15 10:10:29 +00:00
}
_, err := c.client.SetName(&client.SetNameRequest{
FirstName: firstname,
LastName: lastname,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return errors.Wrap(err, "Couldn't set name").Error(), false
2022-08-15 10:10:29 +00:00
}
2019-12-05 18:13:17 +00:00
}
// set About
case "setbio":
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, false
}
2019-12-05 18:13:17 +00:00
_, err := c.client.SetBio(&client.SetBioRequest{
Bio: rawCmdArguments(cmdline, 0),
2019-12-05 18:13:17 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return errors.Wrap(err, "Couldn't set bio").Error(), false
2019-12-05 18:13:17 +00:00
}
// set password
case "setpassword":
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, false
}
2019-12-05 18:13:17 +00:00
var oldPassword string
var newPassword string
// 0 or 1 argument is ignored and the password is reset
if len(args) > 1 {
oldPassword = args[0]
newPassword = args[1]
}
_, err := c.client.SetPassword(&client.SetPasswordRequest{
OldPassword: oldPassword,
NewPassword: newPassword,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return errors.Wrap(err, "Couldn't set password").Error(), false
2019-12-05 18:13:17 +00:00
}
2019-12-05 19:56:12 +00:00
case "config":
if len(args) > 1 {
var msg string
2023-08-02 21:08:06 +00:00
if gateway.MessageOutgoingPermissionVersion == 0 && args[0] == "carbons" && args[1] == "true" {
2024-02-18 09:36:23 +00:00
return "The server did not allow to enable carbons", false
2023-03-18 21:43:11 +00:00
}
2019-12-05 19:56:12 +00:00
value, err := c.Session.Set(args[0], args[1])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
2019-12-05 19:56:12 +00:00
}
2022-01-05 21:04:22 +00:00
gateway.DirtySessions = true
2019-12-05 19:56:12 +00:00
2024-02-18 09:36:23 +00:00
return fmt.Sprintf("%s%s set to %s", msg, args[0], value), true
2019-12-05 19:56:12 +00:00
} else if len(args) > 0 {
value, err := c.Session.Get(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
2019-12-05 19:56:12 +00:00
}
2024-02-18 09:36:23 +00:00
return fmt.Sprintf("%s is set to %s", args[0], value), true
2019-12-05 19:56:12 +00:00
}
var entries []string
for key, value := range c.Session.ToMap() {
entries = append(entries, fmt.Sprintf("%s is set to %s", key, value))
}
2024-02-18 09:36:23 +00:00
return strings.Join(entries, "\n"), true
2022-02-26 19:35:43 +00:00
case "report":
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
2022-02-26 19:35:43 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", false
}
2022-02-26 19:35:43 +00:00
text := rawCmdArguments(cmdline, 1)
_, err = c.client.ReportChat(&client.ReportChatRequest{
2022-03-09 19:16:13 +00:00
ChatId: contact.Id,
2023-11-11 21:10:23 +00:00
Reason: &client.ReportReasonCustom{},
2022-03-09 19:16:13 +00:00
Text: text,
2022-02-26 19:35:43 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
2022-02-26 19:35:43 +00:00
} else {
2024-02-18 09:36:23 +00:00
return "Reported", true
2022-02-26 19:35:43 +00:00
}
case "add":
return c.cmdAdd(args)
case "join":
return c.cmdJoin(args)
case "supergroup":
return c.cmdSupergroup(args, cmdline)
case "channel":
return c.cmdChannel(args, cmdline)
2019-11-26 00:04:11 +00:00
case "help":
2024-02-18 09:36:23 +00:00
return c.helpString(CommandTypeTransport, 0), true
2019-11-24 17:10:29 +00:00
}
2024-02-18 09:36:23 +00:00
return "", true
2019-11-24 17:10:29 +00:00
}
2019-12-03 00:32:53 +00:00
// ProcessChatCommand executes a command sent in a mapped chat
2024-02-18 09:36:23 +00:00
// and returns a response, the status of command support and the execution success result
func (c *Client) ProcessChatCommand(chatID int64, cmdline string) (string, bool, bool) {
if !c.Online() {
2024-02-18 09:36:23 +00:00
return notOnline, true, false
}
2019-12-05 23:21:39 +00:00
cmd, args := parseCommand(cmdline)
2024-02-10 18:46:02 +00:00
command, ok := chatCommands[cmd]
if !ok {
2024-02-18 09:36:23 +00:00
return unknownCommand, false, false
2024-02-10 18:46:02 +00:00
}
if len(args) < command.RequiredArgs {
2024-02-18 09:36:23 +00:00
return notEnoughArguments, true, false
2024-02-10 18:46:02 +00:00
}
2024-02-15 09:40:57 +00:00
chatType, chatTypeErr := c.GetChatType(chatID)
if chatTypeErr == nil && !IsCommandForChatType(command, chatType) {
2024-02-18 09:36:23 +00:00
return "Not applicable for this chat type", true, false
2024-02-15 09:40:57 +00:00
}
2019-12-03 00:32:53 +00:00
switch cmd {
2022-01-06 12:13:57 +00:00
// delete message
2019-12-05 23:21:39 +00:00
case "d":
if c.me == nil {
2024-02-18 09:36:23 +00:00
return "@me is not initialized", true, false
2019-12-05 23:21:39 +00:00
}
2019-12-05 23:21:39 +00:00
var limit int32
if len(args) > 0 {
limit64, err := strconv.ParseInt(args[0], 10, 32)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-05 23:21:39 +00:00
}
limit = int32(limit64)
} else {
limit = 1
}
2022-01-17 20:45:40 +00:00
messages, err := c.getLastMessages(chatID, "", c.me.Id, limit)
2019-12-05 23:21:39 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-05 23:21:39 +00:00
}
log.Debugf("pre-deletion query: %#v %#v", messages, messages.Messages)
var messageIds []int64
for _, message := range messages.Messages {
if message != nil {
2022-01-17 20:45:40 +00:00
messageIds = append(messageIds, message.Id)
}
2019-12-05 23:21:39 +00:00
}
_, err = c.client.DeleteMessages(&client.DeleteMessagesRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
MessageIds: messageIds,
2019-12-05 23:21:39 +00:00
Revoke: true,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-05 23:21:39 +00:00
}
2022-01-06 12:13:57 +00:00
// edit message
2019-12-07 16:37:14 +00:00
case "s":
if c.me == nil {
2024-02-18 09:36:23 +00:00
return "@me is not initialized", true, false
2019-12-07 16:37:14 +00:00
}
2022-01-17 20:45:40 +00:00
messages, err := c.getLastMessages(chatID, "", c.me.Id, 1)
2019-12-07 16:37:14 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-07 16:37:14 +00:00
}
if len(messages.Messages) == 0 {
2024-02-18 09:36:23 +00:00
return "No last message", true, false
2019-12-07 16:37:14 +00:00
}
message := messages.Messages[0]
if message == nil {
2024-02-18 09:36:23 +00:00
return "Last message is empty", true, false
2019-12-07 16:37:14 +00:00
}
2023-06-04 00:25:00 +00:00
content := c.PrepareOutgoingMessageContent(rawCmdArguments(cmdline, 0))
2019-12-07 16:37:14 +00:00
if content != nil {
_, err = c.client.EditMessageText(&client.EditMessageTextRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
MessageId: message.Id,
InputMessageContent: content,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return "Message editing error", true, false
}
} else {
2024-02-18 09:36:23 +00:00
return "Message processing error", true, false
}
// send without sound
case "silent":
2023-06-04 00:25:00 +00:00
content := c.PrepareOutgoingMessageContent(rawCmdArguments(cmdline, 0))
if content != nil {
_, err := c.client.SendMessage(&client.SendMessageRequest{
ChatId: chatID,
InputMessageContent: content,
Options: &client.MessageSendOptions{
DisableNotification: true,
},
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
} else {
2024-02-18 09:36:23 +00:00
return "Message processing error", true, false
}
// schedule a message to timestamp or to going online
case "schedule":
var state client.MessageSchedulingState
var result string
due := args[0]
if due == "online" {
state = &client.MessageSchedulingStateSendWhenOnline{}
result = due
} else {
if c.Session.Timezone == "" {
due += "Z"
} else {
due += c.Session.Timezone
}
switch 0 {
default:
// try bare time first
timestamp, err := time.Parse("15:04:05Z07:00", due)
if err == nil {
now := time.Now().In(c.Session.TimezoneToLocation())
// combine timestamp's time with today's date
timestamp = time.Date(
now.Year(),
now.Month(),
now.Day(),
timestamp.Hour(),
timestamp.Minute(),
timestamp.Second(),
0,
timestamp.Location(),
)
diff := timestamp.Sub(now)
if diff < 0 { // set to tomorrow
timestamp = timestamp.AddDate(0, 0, 1)
}
state = &client.MessageSchedulingStateSendAtDate{
SendDate: int32(timestamp.Unix()),
}
result = timestamp.Format(time.RFC3339)
break
}
timestamp, err = time.Parse(time.RFC3339, due)
if err == nil {
// 2038 doomsday again
state = &client.MessageSchedulingStateSendAtDate{
SendDate: int32(timestamp.Unix()),
}
result = timestamp.Format(time.RFC3339)
break
}
2024-02-18 09:36:23 +00:00
return "Invalid schedule time specifier", true, false
}
}
2023-06-04 00:25:00 +00:00
content := c.PrepareOutgoingMessageContent(rawCmdArguments(cmdline, 1))
if content != nil {
_, err := c.client.SendMessage(&client.SendMessageRequest{
ChatId: chatID,
InputMessageContent: content,
Options: &client.MessageSendOptions{
SchedulingState: state,
},
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2024-02-18 09:36:23 +00:00
return "Scheduled to " + result, true, true
} else {
2024-02-18 09:36:23 +00:00
return "Message processing error", true, false
}
2022-03-12 17:25:53 +00:00
// forward a message to chat
case "forward":
messageId, err := strconv.ParseInt(args[0], 10, 64)
if err != nil {
2024-02-18 09:36:23 +00:00
return "Cannot parse message ID", true, false
2022-03-12 17:25:53 +00:00
}
targetChatParts := strings.Split(args[1], "@") // full JIDs are supported too
targetChatId, err := strconv.ParseInt(targetChatParts[0], 10, 64)
if err != nil {
2024-02-18 09:36:23 +00:00
return "Cannot parse target chat ID", true, false
2022-03-12 17:25:53 +00:00
}
messages, err := c.client.ForwardMessages(&client.ForwardMessagesRequest{
ChatId: targetChatId,
FromChatId: chatID,
MessageIds: []int64{messageId},
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-03-12 17:25:53 +00:00
}
if messages != nil && messages.Messages != nil {
for _, message := range messages.Messages {
c.ProcessIncomingMessage(targetChatId, message)
2022-03-12 17:25:53 +00:00
}
}
2023-06-01 20:37:38 +00:00
// print vCard
case "vcard":
info, err := c.GetVcardInfo(chatID)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2023-06-01 20:37:38 +00:00
}
_, link := c.PermastoreFile(info.Photo, true)
entries := []string{
keyValueString("Chat title", info.Fn),
keyValueString("Photo", link),
keyValueString("Usernames", c.usernamesToString(info.Nicknames)),
2023-06-08 17:33:22 +00:00
keyValueString("Full name", info.Given+" "+info.Family),
2023-06-01 20:37:38 +00:00
keyValueString("Phone number", info.Tel),
}
2024-02-18 09:36:23 +00:00
return strings.Join(entries, "\n"), true, true
2019-12-07 17:56:53 +00:00
// add @contact
case "add":
2024-02-18 09:36:23 +00:00
response, success := c.cmdAdd(args)
return response, true, success
2022-06-22 18:03:18 +00:00
// join https://t.me/publichat or @publicchat
2019-12-07 21:08:12 +00:00
case "join":
2024-02-18 09:36:23 +00:00
response, success := c.cmdJoin(args)
return response, true, success
2019-12-07 21:26:58 +00:00
// create new supergroup
case "supergroup":
2024-02-18 09:36:23 +00:00
response, success := c.cmdSupergroup(args, cmdline)
return response, true, success
2019-12-07 21:26:58 +00:00
// create new channel
case "channel":
2024-02-18 09:36:23 +00:00
response, success := c.cmdChannel(args, cmdline)
return response, true, success
2019-12-07 23:36:29 +00:00
// create new secret chat with current user
case "secret":
2022-01-17 19:58:16 +00:00
_, err := c.client.CreateNewSecretChat(&client.CreateNewSecretChatRequest{
2022-01-17 20:45:40 +00:00
UserId: chatID,
2019-12-07 23:36:29 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-07 23:36:29 +00:00
}
// create group chat with current user
case "group":
2022-01-17 19:58:16 +00:00
_, err := c.client.CreateNewBasicGroupChat(&client.CreateNewBasicGroupChatRequest{
2022-01-17 20:45:40 +00:00
UserIds: []int64{chatID},
2022-01-17 19:58:16 +00:00
Title: args[0],
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-07 23:36:29 +00:00
}
2019-12-08 01:54:09 +00:00
// blacklists current user
case "block":
2023-11-11 21:10:23 +00:00
_, err := c.client.SetMessageSenderBlockList(&client.SetMessageSenderBlockListRequest{
2022-01-17 20:45:40 +00:00
SenderId: &client.MessageSenderUser{UserId: chatID},
2023-11-11 21:10:23 +00:00
BlockList: &client.BlockListMain{},
2022-01-17 19:58:16 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 01:54:09 +00:00
}
// unblacklists current user
case "unblock":
2023-11-11 21:10:23 +00:00
_, err := c.client.SetMessageSenderBlockList(&client.SetMessageSenderBlockListRequest{
2022-01-17 20:45:40 +00:00
SenderId: &client.MessageSenderUser{UserId: chatID},
2023-11-11 21:10:23 +00:00
BlockList: nil,
2022-01-17 19:58:16 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 01:54:09 +00:00
}
2019-12-08 13:32:43 +00:00
// invite @username to current groupchat
case "invite":
2022-01-17 19:58:16 +00:00
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-01-17 19:58:16 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", true, false
}
2019-12-08 13:32:43 +00:00
2022-01-17 19:58:16 +00:00
_, err = c.client.AddChatMember(&client.AddChatMemberRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
UserId: contact.Id,
2022-01-17 19:58:16 +00:00
ForwardLimit: 100,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 13:58:17 +00:00
}
// get link to current chat
case "link":
link, err := c.client.CreateChatInviteLink(&client.CreateChatInviteLinkRequest{
ChatId: chatID,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2024-02-18 09:36:23 +00:00
return link.InviteLink, true, true
2019-12-08 13:58:17 +00:00
// kick @username from current group chat
case "kick":
2022-01-17 19:58:16 +00:00
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-01-17 19:58:16 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", true, false
}
2019-12-08 13:58:17 +00:00
2022-01-17 19:58:16 +00:00
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
MemberId: &client.MessageSenderUser{UserId: contact.Id},
2022-02-01 04:57:17 +00:00
Status: &client.ChatMemberStatusLeft{},
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-02-01 04:57:17 +00:00
}
// mute [@username [n hours]]
2022-02-01 04:57:17 +00:00
case "mute":
if len(args) > 0 {
contact, _, err := c.GetContactByUsername(args[0])
2022-02-01 04:57:17 +00:00
if err != nil {
2024-05-09 23:32:57 +00:00
return err.Error(), true, false
}
if contact == nil {
return "Contact not found", true, false
2022-02-01 04:57:17 +00:00
}
var hours int64
if len(args) > 1 {
hours, err = strconv.ParseInt(args[1], 10, 32)
if err != nil {
2024-05-09 23:32:57 +00:00
return "Invalid number of hours", true, false
}
}
2022-02-01 04:57:17 +00:00
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
ChatId: chatID,
MemberId: &client.MessageSenderUser{UserId: contact.Id},
Status: &client.ChatMemberStatusRestricted{
IsMember: true,
RestrictedUntilDate: c.formatBantime(hours),
Permissions: &permissionsReadonly,
},
})
2022-02-01 04:57:17 +00:00
if err != nil {
2024-05-09 23:32:57 +00:00
return err.Error(), true, false
2022-02-01 04:57:17 +00:00
}
} else {
if !c.Session.IgnoreChat(chatID) {
2024-05-09 23:32:57 +00:00
return "Chat is already ignored", true, false
}
gateway.DirtySessions = true
2022-02-01 04:57:17 +00:00
}
// unmute [@username]
2022-02-01 04:57:17 +00:00
case "unmute":
if len(args) > 0 {
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-05-09 23:32:57 +00:00
return err.Error(), true, false
}
if contact == nil {
return "Contact not found", true, false
}
2022-02-01 04:57:17 +00:00
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
ChatId: chatID,
MemberId: &client.MessageSenderUser{UserId: contact.Id},
Status: &client.ChatMemberStatusRestricted{
IsMember: true,
RestrictedUntilDate: 0,
Permissions: &permissionsMember,
},
})
if err != nil {
2024-05-09 23:32:57 +00:00
return err.Error(), true, false
}
} else {
if !c.Session.UnignoreChat(chatID) {
2024-05-09 23:32:57 +00:00
return "Chat wasn't ignored", true, false
}
gateway.DirtySessions = true
2019-12-08 13:32:43 +00:00
}
2019-12-08 14:24:51 +00:00
// ban @username from current chat [for N hours]
case "ban":
2022-01-17 19:58:16 +00:00
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-01-17 19:58:16 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", true, false
}
2019-12-08 14:24:51 +00:00
2022-01-17 19:58:16 +00:00
var hours int64
if len(args) > 1 {
hours, err = strconv.ParseInt(args[1], 10, 32)
2019-12-08 14:24:51 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return "Invalid number of hours", true, false
2019-12-08 14:24:51 +00:00
}
}
2022-01-17 19:58:16 +00:00
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
2022-02-08 20:25:58 +00:00
ChatId: chatID,
2022-01-17 20:45:40 +00:00
MemberId: &client.MessageSenderUser{UserId: contact.Id},
2022-02-01 04:57:17 +00:00
Status: &client.ChatMemberStatusBanned{
BannedUntilDate: c.formatBantime(hours),
},
2022-01-17 19:58:16 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-01-17 19:58:16 +00:00
}
2022-02-03 18:51:27 +00:00
// unban @username
case "unban":
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-02-03 18:51:27 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", true, false
}
2022-02-03 18:51:27 +00:00
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
2022-02-08 20:25:58 +00:00
ChatId: chatID,
2022-02-03 18:51:27 +00:00
MemberId: &client.MessageSenderUser{UserId: contact.Id},
2022-02-08 20:25:58 +00:00
Status: &client.ChatMemberStatusMember{},
2022-02-03 18:51:27 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-02-03 18:51:27 +00:00
}
// promote @username to admin
case "promote":
contact, _, err := c.GetContactByUsername(args[0])
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-02-03 18:51:27 +00:00
}
if contact == nil {
2024-02-18 09:36:23 +00:00
return "Contact not found", true, false
}
2022-02-03 18:51:27 +00:00
// clone the permissions
status := client.ChatMemberStatusAdministrator{
CanBeEdited: true,
Rights: &permissionsAdmin,
}
2022-02-03 18:51:27 +00:00
if len(args) > 1 {
status.CustomTitle = args[1]
}
_, err = c.client.SetChatMemberStatus(&client.SetChatMemberStatusRequest{
2022-02-08 20:25:58 +00:00
ChatId: chatID,
2022-02-03 18:51:27 +00:00
MemberId: &client.MessageSenderUser{UserId: contact.Id},
2022-02-08 20:25:58 +00:00
Status: &status,
2022-02-03 18:51:27 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2022-02-03 18:51:27 +00:00
}
2019-12-08 15:08:55 +00:00
// leave current chat
case "leave":
2022-01-17 19:58:16 +00:00
_, err := c.client.LeaveChat(&client.LeaveChatRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
2022-01-17 19:58:16 +00:00
})
2019-12-08 15:08:55 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 15:08:55 +00:00
}
err = c.unsubscribe(chatID)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
// leave current chat (for owners)
case "leave!":
_, err := c.client.DeleteChat(&client.DeleteChatRequest{
ChatId: chatID,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
err = c.unsubscribe(chatID)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
// set TTL
case "ttl":
var ttl int64
var err error
if len(args) > 0 {
ttl, err = strconv.ParseInt(args[0], 10, 32)
if err != nil {
2024-02-18 09:36:23 +00:00
return "Invalid TTL", true, false
}
}
_, err = c.client.SetChatMessageAutoDeleteTime(&client.SetChatMessageAutoDeleteTimeRequest{
2023-08-01 01:37:05 +00:00
ChatId: chatID,
MessageAutoDeleteTime: int32(ttl),
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2019-12-08 15:25:29 +00:00
// close secret chat
case "close":
chat, _, err := c.GetContactByID(chatID, nil)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 15:25:29 +00:00
}
if chat == nil {
2024-02-18 09:36:23 +00:00
return "Chat not found", true, false
}
2019-12-08 15:25:29 +00:00
chatType := chat.Type.ChatTypeType()
if chatType == client.TypeChatTypeSecret {
chatTypeSecret, _ := chat.Type.(*client.ChatTypeSecret)
_, err = c.client.CloseSecretChat(&client.CloseSecretChatRequest{
2022-01-17 20:45:40 +00:00
SecretChatId: chatTypeSecret.SecretChatId,
2019-12-08 15:25:29 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 15:25:29 +00:00
}
err = c.unsubscribe(chatID)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2019-12-08 15:08:55 +00:00
}
2019-12-08 15:35:27 +00:00
// delete current chat
case "delete":
_, err := c.client.DeleteChatHistory(&client.DeleteChatHistoryRequest{
2022-01-17 20:45:40 +00:00
ChatId: chatID,
2019-12-08 15:35:27 +00:00
RemoveFromChatList: true,
2022-01-17 19:58:16 +00:00
Revoke: true,
2019-12-08 15:35:27 +00:00
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 15:35:27 +00:00
}
err = c.unsubscribe(chatID)
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2022-01-06 12:13:57 +00:00
// message search
2019-12-08 16:04:26 +00:00
case "search":
2022-01-17 19:58:16 +00:00
var limit int32 = 100
2019-12-08 16:04:26 +00:00
if len(args) > 1 {
newLimit, err := strconv.ParseInt(args[1], 10, 32)
if err == nil {
limit = int32(newLimit)
}
}
var query string
if len(args) > 0 {
query = args[0]
}
2022-01-06 12:13:57 +00:00
messages, err := c.getLastMessages(chatID, query, 0, limit)
2019-12-08 16:04:26 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 16:04:26 +00:00
}
2019-12-08 16:19:35 +00:00
c.sendMessagesReverse(chatID, messages.Messages)
// get latest entries from history
case "history":
var limit int32 = 10
if len(args) > 0 {
newLimit, err := strconv.ParseInt(args[0], 10, 32)
if err == nil {
limit = int32(newLimit)
}
}
2022-05-18 14:18:47 +00:00
var newMessages *client.Messages
var messages []*client.Message
var err error
2022-05-18 14:18:47 +00:00
var fromId int64
for _ = range make([]struct{}, limit) { // safety limit
if len(messages) > 0 {
fromId = messages[len(messages)-1].Id
}
newMessages, err = c.client.GetChatHistory(&client.GetChatHistoryRequest{
ChatId: chatID,
FromMessageId: fromId,
Limit: limit,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
}
2022-05-18 14:18:47 +00:00
messages = append(messages, newMessages.Messages...)
if len(newMessages.Messages) == 0 || len(messages) >= int(limit) {
break
}
2019-12-08 16:04:26 +00:00
}
2019-12-08 16:19:35 +00:00
2022-05-18 14:18:47 +00:00
c.sendMessagesReverse(chatID, messages)
2022-01-17 19:58:16 +00:00
// chat members
2019-12-08 18:44:17 +00:00
case "members":
var query string
if len(args) > 0 {
query = args[0]
}
members, err := c.GetChatMembers(chatID, false, query, MembersListMembers)
2019-12-08 18:44:17 +00:00
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), true, false
2019-12-08 18:44:17 +00:00
}
var entries []string
for _, member := range members {
senderId := c.GetSenderId(member.MemberId)
2019-12-08 18:44:17 +00:00
entries = append(entries, fmt.Sprintf(
"%v | role: %v",
c.FormatContact(senderId),
2019-12-08 18:44:17 +00:00
member.Status.ChatMemberStatusType(),
))
}
2024-02-18 09:36:23 +00:00
return strings.Join(entries, "\n"), true, true
2019-12-03 00:32:53 +00:00
case "help":
2024-02-18 09:36:23 +00:00
return c.helpString(CommandTypeChat, chatID), true, true
2019-12-05 23:21:39 +00:00
default:
2024-02-18 09:36:23 +00:00
return "", false, false
2019-12-03 00:32:53 +00:00
}
2024-02-18 09:36:23 +00:00
return "", true, true
2019-12-03 00:32:53 +00:00
}
2024-02-18 09:36:23 +00:00
func (c *Client) cmdAdd(args []string) (string, bool) {
chat, err := c.client.SearchPublicChat(&client.SearchPublicChatRequest{
Username: args[0],
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
if chat == nil {
2024-02-18 09:36:23 +00:00
return "No error, but chat is nil", false
}
c.subscribeToID(chat.Id, chat)
2024-02-18 09:36:23 +00:00
return "", true
}
2024-02-18 09:36:23 +00:00
func (c *Client) cmdJoin(args []string) (string, bool) {
if strings.HasPrefix(args[0], "@") {
chat, err := c.client.SearchPublicChat(&client.SearchPublicChatRequest{
Username: args[0],
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
if chat == nil {
2024-02-18 09:36:23 +00:00
return "No error, but chat is nil", false
}
_, err = c.client.JoinChat(&client.JoinChatRequest{
ChatId: chat.Id,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
} else {
_, err := c.client.JoinChatByInviteLink(&client.JoinChatByInviteLinkRequest{
InviteLink: args[0],
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
}
2024-02-18 09:36:23 +00:00
return "", true
}
2024-02-18 09:36:23 +00:00
func (c *Client) cmdSupergroup(args []string, cmdline string) (string, bool) {
_, err := c.client.CreateNewSupergroupChat(&client.CreateNewSupergroupChatRequest{
Title: args[0],
Description: rawCmdArguments(cmdline, 1),
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
2024-02-18 09:36:23 +00:00
return "", true
}
2024-02-18 09:36:23 +00:00
func (c *Client) cmdChannel(args []string, cmdline string) (string, bool) {
_, err := c.client.CreateNewSupergroupChat(&client.CreateNewSupergroupChatRequest{
Title: args[0],
Description: rawCmdArguments(cmdline, 1),
IsChannel: true,
})
if err != nil {
2024-02-18 09:36:23 +00:00
return err.Error(), false
}
2024-02-18 09:36:23 +00:00
return "", true
}