/* * @Authors: Amalvy Arthur, Bartuccio Antoine */ // This module uses a property named "subscribed_chats" in the shared Users structure. // It consists of a list of chatID as a string, chatID are separated by ":" (unix PATH style) // This module uses a property named "saved_messages" in the shared Chats structure // It consists of a slice of string package commands import ( "encoding/json" "strconv" "strings" "../shared" tb "gopkg.in/tucnak/telebot.v2" ) func getPublishedMessages(chatID int64) []tb.Message { messages := []tb.Message{} serializedMessages, exists := shared.ChatData.Get(chatID, "published_messages") if !exists { return messages } if _, ok := serializedMessages.(string); !ok { return messages } if json.Unmarshal([]byte(serializedMessages.(string)), &messages) != nil { return []tb.Message{} } return messages } func setPublishedMessages(chatID int64, messages []tb.Message) { data, err := json.Marshal(messages) if err != nil { return } shared.ChatData.Set(chatID, "published_messages", string(data)) } // Subscribe user sending the command to the current chat // Command syntax : /subscribe func Subscribe(m *tb.Message) { if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } if m.Sender.Username == "" { shared.Bot.Send(m.Chat, "Il faut avoir enregistré un username pour pouvoir utiliser cette fonction") return } userSubscribedChats, exists := shared.Users.Get(m.Sender.Username, "subscribed_chats") if !exists { shared.Bot.Send(m.Chat, "Abonnement au chat : "+m.Chat.Title) shared.Users.Set(m.Sender.Username, "subscribed_chats", strconv.FormatInt(m.Chat.ID, 10)) return } splittedChats := strings.Split(userSubscribedChats, ":") for _, chatID := range splittedChats { if chatID == strconv.FormatInt(m.Chat.ID, 10) { shared.Bot.Send(m.Chat, "Vous êtes déjà abonné à ce chat : "+m.Chat.Title) return } } shared.Bot.Send(m.Chat, "Abonnement au chat : "+m.Chat.Title) if len(userSubscribedChats) != 0 { shared.Users.Set(m.Sender.Username, "subscribed_chats", userSubscribedChats+":"+strconv.FormatInt(m.Chat.ID, 10)) } else { shared.Users.Set(m.Sender.Username, "subscribed_chats", strconv.FormatInt(m.Chat.ID, 10)) } } // Unsubscribe user sending the command from the current chat // Command syntax : /unsubscribe func Unsubscribe(m *tb.Message) { if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } if m.Sender.Username == "" { shared.Bot.Send(m.Chat, "Il faut avoir enregistré un username pour pouvoir utiliser cette fonction") return } userSubscribedChats, exists := shared.Users.Get(m.Sender.Username, "subscribed_chats") if !exists || len(userSubscribedChats) == 0 { shared.Bot.Send(m.Chat, "Vous n'êtes abonné à aucun chat") return } splittedChats := strings.Split(userSubscribedChats, ":") for i, chatID := range splittedChats { if chatID == strconv.FormatInt(m.Chat.ID, 10) { shared.Bot.Send(m.Chat, "désabonnement du chat : "+m.Chat.Title) splittedChats = append(splittedChats[:i], splittedChats[i+1:]...) break } } shared.Users.Set(m.Sender.Username, "subscribed_chats", strings.Join(splittedChats, ":")) } // ListSubscribers List all subscribers of the current chat // Command syntax : /listsubscribers func ListSubscribers(m *tb.Message) { if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } subscribers := m.Chat.Title + " subscribers : \n\n" for _, subscriber := range getSubscribers(m.Chat.ID) { subscribers = subscribers + "- " + subscriber + "\n" } shared.Bot.Send(m.Chat, subscribers) } // Publish a message from the current chat // Command syntax (while replying to a message) : /publish func Publish(m *tb.Message) { if m.ReplyTo == nil { shared.Bot.Send(m.Chat, "Veuillez répondre à un message pour le publier") return } if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } defer shared.Bot.Send(m.Chat, "Message publié : "+m.ReplyTo.Text) savedMessages := getPublishedMessages(m.Chat.ID) setPublishedMessages(m.Chat.ID, append(savedMessages, *m.ReplyTo)) } // Unpublish remove a message from published messages in the current chat // Command syntax (while replying to a message) : /unpublish func Unpublish(m *tb.Message) { if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } if m.ReplyTo == nil { shared.Bot.Send(m.Chat, "Veuillez répondre à un message pour le dépublier") return } if !m.ReplyTo.IsForwarded() { shared.Bot.Send(m.Chat, "Ce message ne peut pas avoir été publié") return } publishedMessages := getPublishedMessages(m.Chat.ID) filteredPublishedMessages := []tb.Message{} found := false for _, message := range publishedMessages { // You can't just compare messages id because // when retrieving, the newly send message // has a different ID from the one stored so you can't detect that's // it's in the database except if you find the original message // which is very unlikely to happen // As a workaround, we check the original sender, original unix time // and associated text if message.Sender.ID == m.ReplyTo.OriginalSender.ID && message.Unixtime == int64(m.ReplyTo.OriginalUnixtime) && message.Text == m.ReplyTo.Text { found = true continue } filteredPublishedMessages = append(filteredPublishedMessages, message) } if !found { shared.Bot.Send(m.Chat, "Ce message n'a jamais été publié") return } setPublishedMessages(m.Chat.ID, filteredPublishedMessages) shared.Bot.Send(m.Chat, "Message supprimé des publication") } // Retrieve If performed in MP : retrieve all messages from all subscribed sources for the user // If performed in Group Chat : retrieved all published messages for this chat // Command syntax : /retrieve func Retrieve(m *tb.Message) { chatList := []int64{} hasMessage := false if m.Chat.Type != tb.ChatGroup && m.Chat.Type != tb.ChatSuperGroup && m.Chat.Type != tb.ChatPrivate { shared.Bot.Send(m.Chat, "Cette commande n'est pas autorisée pour ce type de chat") return } if m.Chat.Type == tb.ChatPrivate { userSubscribedChats, exists := shared.Users.Get(m.Sender.Username, "subscribed_chats") if !exists || len(userSubscribedChats) == 0 { shared.Bot.Send(m.Chat, "Aucun abonnements") return } for _, chat := range strings.Split(userSubscribedChats, ":") { chatID, err := strconv.ParseInt(chat, 10, 64) if err != nil { shared.Bot.Send(m.Chat, "Erreur lors de la récupération de vos évènements") return } chatList = append(chatList, chatID) } } if m.Chat.Type == tb.ChatGroup || m.Chat.Type == tb.ChatSuperGroup { chatList = append(chatList, m.Chat.ID) } shared.Bot.Send(m.Chat, "--- Messages publiés ---") for _, chatID := range chatList { messages := getPublishedMessages(chatID) if len(messages) > 0 { hasMessage = true } for _, message := range messages { shared.Bot.Forward(m.Chat, &message) } } if !hasMessage { shared.Bot.Send(m.Chat, "Aucun message publié") } shared.Bot.Send(m.Chat, "--- Messages publiés ---") } // Get all users subscribed to the provided channel func getSubscribers(chatID int64) []string { var subscribers []string for _, username := range shared.Users.GetUsernames() { userSubscribedChats, exists := shared.Users.Get(username, "subscribed_chats") if exists { splittedChats := strings.Split(userSubscribedChats, ":") for _, splittedChatID := range splittedChats { if splittedChatID == strconv.FormatInt(chatID, 10) { subscribers = append(subscribers, username) } } } } return subscribers }