diff --git a/main.go b/main.go new file mode 100644 index 0000000..71c60d5 --- /dev/null +++ b/main.go @@ -0,0 +1,843 @@ +package main + +import ( + "fmt" + "regexp" + "strconv" + "strings" + "sync" + "time" + "crypto/md5" + "encoding/base32" + "encoding/json" + "math/rand" + + "github.com/Arman92/go-tdlib" + "github.com/streadway/amqp" +// "github.com/go-web/tokenizer" + +// "database/sql" +// _ "github.com/go-sql-driver/mysql" +) + +type MsgDirection int + +const ( + Incoming MsgDirection = 1 + Outgoing MsgDirection = 2 +) + +type ChatWarsMessage struct { + UserID64 int64 `json:"user_id"` + Direction MsgDirection `json:"direction"` + MsgText string `json:"msg"` + MsgID64 int64 `json:"msg_id"` +} + +type ChatWarsWithdraw struct { + Item string `json:"item"` + Quantity int64 `json:"quantity"` + UserID64 int64 `json:"user_id"` + Time time.Time `json:"time"` +} + +const user_chtwrsbot = 408101137 +const user_botnestbot = 541634211 +const user_chtwrscastlebot = 526586204 +const chat_darkwing = -1001152534174 +const chat_guild_deposit = -1001179598662 +const chat_guild = -1001152534174 +const chat_private = -1001092914472 + +var ownUserID64 = int64(0) +var ownUserID32 = int32(0) +var msgMutex sync.Mutex +var msgID64 = int64(0) +var msgUserID64 = int64(0) +var msgText string +var vault Vault +var size int +var withdrawItem string +var withdrawUserID64 int64 +var withdrawTime time.Time +var aesKey []byte +var hmacKey []byte +var withdrawals = map[string]ChatWarsWithdraw{} +var TgEncoding *base32.Encoding + + +func main() { +// msgMutex = &sync.Mutex{} + tdlib.SetLogVerbosityLevel(1) + tdlib.SetFilePath("./errors.txt") + + TgEncoding = base32.StdEncoding.WithPadding(base32.NoPadding) + + + // Create new instance of client + client := tdlib.NewClient(tdlib.Config{ + APIID: "187786", + APIHash: "e782045df67ba48e441ccb105da8fc85", + SystemLanguageCode: "en", + DeviceModel: "Shoogram", + SystemVersion: "1.3.3.7", + ApplicationVersion: "3.1.3.3.7", + UseMessageDatabase: true, + UseFileDatabase: true, + UseChatInfoDatabase: true, + UseTestDataCenter: false, + DatabaseDirectory: "./tdlib-db", + FileDirectory: "./tdlib-files", + IgnoreFileNames: false, + }) + + for { + currentState, _ := client.Authorize() + if currentState.GetAuthorizationStateEnum() == tdlib.AuthorizationStateWaitPhoneNumberType { + fmt.Print("Enter phone: ") + var number string + fmt.Scanln(&number) + _, err := client.SendPhoneNumber(number) + if err != nil { + fmt.Printf("Error sending phone number: %v", err) + } + } else if currentState.GetAuthorizationStateEnum() == tdlib.AuthorizationStateWaitCodeType { + fmt.Print("Enter code: ") + var code string + fmt.Scanln(&code) + _, err := client.SendAuthCode(code) + if err != nil { + fmt.Printf("Error sending auth code : %v", err) + } + } else if currentState.GetAuthorizationStateEnum() == tdlib.AuthorizationStateWaitPasswordType { + fmt.Print("Enter Password: ") + var password string + fmt.Scanln(&password) + _, err := client.SendAuthPassword(password) + if err != nil { + fmt.Printf("Error sending auth password: %v", err) + } + } else if currentState.GetAuthorizationStateEnum() == tdlib.AuthorizationStateReadyType { + fmt.Println("Authorization Ready! Let's rock") + break + } + } + + ownUserID32 = OwnUserID(client) + ownUserID64 = int64(OwnUserID(client)) + + go PillageKnight(client) + go ListenDarkwing(client) + go ListenCW(client) + go Listen(client) + go ListenGuild(client) +// go ForwardOwnMsg(client) + + fmt.Println("Started !") + + // Main loop + for { + time.Sleep(30 * time.Second) + } +} + +func PillageKnight(c *tdlib.Client) { + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID + senderUserID := updateMsg.Message.SenderUserID +// viaBotUserID := updateMsg.Message.ViaBotUserID +// fmt.Println("UpdateNewMessage : chatID : ", chatID) +// fmt.Println("UpdateNewMessage : senderUserID : ", senderUserID) +// fmt.Println("UpdateNewMessage : viaBotUserID : ", viaBotUserID) + if (senderUserID == user_chtwrsbot && chatID == user_chtwrsbot) { + return true + } + return false + } + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) +// senderUserID := updateMsg.Message.SenderUserID + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { +/* + user, err := c.GetUser(senderUserID) + if err != nil { + fmt.Println("Pillage:", err.Error()) + continue + } +*/ + txt := updateMsg.Message.Content.(*tdlib.MessageText).Text.Text + + for _, line := range strings.Split(strings.TrimSuffix(txt, "\n"), "\n") { + match, _ := regexp.MatchString("You were strolling around on your horse.*To stop him click /go.*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! PILLAGE KNIGHT INCOMING !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + fwdMsg := ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, ownUserID64) + delay := rand.Intn(20) + fmt.Println("Delaying reply by", delay + 5, "seconds") + time.Sleep(time.Duration(delay) * time.Second) + c.SetOption("online", tdlib.NewOptionValueBoolean(true)) + time.Sleep(5 * time.Second) + + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText("/go\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ := c.SendMessage(updateMsg.Message.ChatID, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/go") + fmt.Println("send msgID64 (go) : ", msgID64) + + delay = rand.Intn(10) + 20 + time.Sleep(time.Duration(delay) * time.Second) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("πŸ›‘Defend\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(updateMsg.Message.ChatID, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("πŸ›‘Defend") + fmt.Println("send msgID64 (defend) : ", msgID64) + + time.Sleep(1 * time.Second) + DeleteMsg(c, ownUserID64, fwdMsg) + time.Sleep(7 * time.Second) +// c.SetOption("online", tdlib.NewOptionValueBoolean(false)) + break + } + match, _ = regexp.MatchString(".*/pledge.*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! PLEDGE INCOMING !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + fwdMsg := ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, ownUserID64) + delay := rand.Intn(20) + fmt.Println("Delaying reply by", delay, "seconds") + time.Sleep(time.Duration(delay) * time.Second) + c.SetOption("online", tdlib.NewOptionValueBoolean(true)) + time.Sleep(5 * time.Second) + + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText("/pledge\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ := c.SendMessage(updateMsg.Message.ChatID, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/pledge") + fmt.Println("send msgID64 (pledge) : ", msgID64) + + time.Sleep(1 * time.Second) + DeleteMsg(c, ownUserID64, fwdMsg) + time.Sleep(7 * time.Second) +// c.SetOption("online", tdlib.NewOptionValueBoolean(false)) + break + } + } + } + } +} + +func ListenCW(c *tdlib.Client) { + var msgDisplay, msgDelete bool + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID + senderUserID := updateMsg.Message.SenderUserID + if (senderUserID == user_chtwrsbot && chatID == user_chtwrsbot) { + return true + } + return false + } + + conn, _ := amqp.Dial("amqp://shoopea:UmDd5g4WRa2MzqOHsG2T@localhost:5672/chatwars") + defer conn.Close() + + ch, _ := conn.Channel() + defer ch.Close() + + q, _ := ch.QueueDeclare( + "msg", // name + false, // durable + false, // delete when unused + false, // exclusive + false, // no-wait + nil, // arguments + ) + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) + senderUserID := updateMsg.Message.SenderUserID + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { + user, err := c.GetUser(senderUserID) + if err != nil { + fmt.Println("ListenCW:", err.Error()) + continue + } + txt := updateMsg.Message.Content.(*tdlib.MessageText).Text.Text + fmt.Println("============== CHAT WARS (", user.Username, ") ================================================================") + fmt.Println(txt, "\n") + fmt.Println("===============================================================================================================") + + m := ChatWarsMessage{ + UserID64 : ownUserID64, + MsgText : string(txt), + Direction : Incoming, + MsgID64 : updateMsg.Message.ID, + } + + b, err := json.Marshal(m) + + _ = ch.Publish( + "", // exchange + q.Name, // routing key + false, // mandatory + false, // immediate + amqp.Publishing{ + ContentType: "application/json", + Body: []byte(b), + }) + + fmt.Println("recv msgID64 : ", msgID64) + if msgID64 != int64(0) { + switch msgText { + case "/g_stock_res": vault.Res = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_stock_alch": vault.Alch = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_stock_misc": vault.Misc = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_stock_rec": vault.Rec = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_stock_parts": vault.Parts = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_stock_other": vault.Other = string(txt) ; msgDisplay = false; msgDelete = true + case "/g_withdraw": time.Sleep(1 * time.Second) ; _ = ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, msgUserID64) ; msgDisplay = false; msgDelete = false + default: fmt.Println("No specific task for ", msgText) ; msgDisplay = true; msgDelete = false + } + time.Sleep(1 * time.Second) + if msgDelete { + DeleteMsg(c, user_chtwrsbot, msgID64) + time.Sleep(1 * time.Second) + DeleteMsg(c, user_chtwrsbot, updateMsg.Message.ID) + } + msgID64 = int64(0) + msgUserID64 = int64(0) + msgMutex.Unlock() + } else { + msgDisplay = true + } + + if msgDisplay { + for _, line := range strings.Split(strings.TrimSuffix(txt, "\n"), "\n") { +// fmt.Println("Testing line:", line) + match, _ := regexp.MatchString(".*Your result on the battlefield:.*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! REPORT SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(user_botnestbot)) + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(user_chtwrscastlebot)) + break + } + match, _ = regexp.MatchString(".*Leaderboard of fighters are updated: /top5 & /top6. ((Cheer up)|(Congratulations))!.*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! FIGHT SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(user_chtwrscastlebot)) + break + } + match, _ = regexp.MatchString("Deposited [βš‘πŸ“•πŸ“—πŸ“˜πŸ“™πŸ“’'a-zA-Z0-9+ ]* \\([0-9]*\\) successfully", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! DEPOSIT SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(chat_guild_deposit)) + break + } + match, _ = regexp.MatchString("Received [βš‘πŸ“•πŸ“—πŸ“˜πŸ“™πŸ“’'a-zA-Z0-9+ ]* x [0-9]*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! RECEPTION SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(chat_guild_deposit)) + break + } + } + } + } + } +} + +func ListenDarkwing(c *tdlib.Client) { + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID + senderUserID := updateMsg.Message.SenderUserID + if (senderUserID == user_botnestbot && chatID == chat_darkwing) { + return true + } + return false + } + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) + senderUserID := updateMsg.Message.SenderUserID + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { + user, err := c.GetUser(senderUserID) + if err != nil { + fmt.Println("ListenDarkwing:", err.Error()) + continue + } + txt := updateMsg.Message.Content.(*tdlib.MessageText).Text.Text + fmt.Println("============== DARKWING (", user.Username, ") =================================================================") + fmt.Println(txt, "\n") + fmt.Println("===============================================================================================================") + + for _, line := range strings.Split(strings.TrimSuffix(txt, "\n"), "\n") { + match, _ := regexp.MatchString(".*Your result on the battlefield:.*", line) + if match { + fmt.Println("!!!!!!!!!!!!!!!!! REPORT SPOTTED !!!!!!!!!!!!!!!!!") + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(user_botnestbot)) + time.Sleep(1 * time.Second) + ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, int64(user_chtwrscastlebot)) + break + } + } + } + } +} + +func ListenGuild(c *tdlib.Client) { + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID +// senderUserID := updateMsg.Message.SenderUserID + if (chatID == chat_guild) { + return true + } + return false + } + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) + senderUserID := updateMsg.Message.SenderUserID + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { + user, err := c.GetUser(senderUserID) + if err != nil { + fmt.Println("ListenGuild:", err.Error()) + continue + } + txt := updateMsg.Message.Content.(*tdlib.MessageText).Text.Text + fmt.Println("============== GUILD (", user.Username, ") =================================================================") + fmt.Println(txt, "\n") + fmt.Println("===============================================================================================================") + + for _, line := range strings.Split(strings.TrimSuffix(txt, "\n"), "\n") { + validID := regexp.MustCompile(`^!g_stock$`) + if validID.MatchString(line) { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! GUILD STOCK REPORT SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_res\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ := c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_res") + fmt.Println("send msgID64 (g_stock_res) : ", msgID64) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_alch\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_alch") + fmt.Println("send msgID64 (g_stock_alch) : ", msgID64) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_misc\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_misc") + fmt.Println("send msgID64 (g_stock_misc) : ", msgID64) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_rec\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_rec") + fmt.Println("send msgID64 (g_stock_rec) : ", msgID64) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_parts\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_parts") + fmt.Println("send msgID64 (g_stock_parts) : ", msgID64) + + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText("/g_stock_other\n", nil), true, true) + msgMutex.Lock() + sendMsg, _ = c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_stock_other") + fmt.Println("send msgID64 (g_stock_other) : ", msgID64) + + msgMutex.Lock() + msgMutex.Unlock() + + r := regexp.MustCompile(`(?P[a-z0-9]+) (?P[βš‘πŸ“•πŸ“—πŸ“˜πŸ“™πŸ“’'a-zA-Z0-9+ ]+) x (?P[0-9]+)`) + s := regexp.MustCompile(`(?P⚑\+[0-9]+) (?P[a-zA-Z ]+)`) + + fmt.Println("=====================> Exploring vault.Res") + vault.ResSize = 0 + vault.ResCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Res, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.ResSize = vault.ResSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.ResCount = vault.ResCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Res : %d\n", vault.ResSize) + + fmt.Println("=====================> Exploring vault.Alch") + vault.AlchSize = 0 + vault.AlchCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Alch, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.AlchSize = vault.AlchSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.AlchCount = vault.AlchCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Alch : %d\n", vault.AlchSize) + + fmt.Println("=====================> Exploring vault.Misc") + vault.MiscSize = 0 + vault.MiscCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Misc, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.MiscSize = vault.MiscSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.MiscCount = vault.MiscCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Misc : %d\n", vault.MiscSize) + + fmt.Println("=====================> Exploring vault.Rec") + vault.RecSize = 0 + vault.RecCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Rec, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.RecSize = vault.RecSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.RecCount = vault.RecCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Rec : %d\n", vault.RecSize) + + fmt.Println("=====================> Exploring vault.Parts") + vault.PartsSize = 0 + vault.PartsCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Parts, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.PartsSize = vault.PartsSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.PartsCount = vault.PartsCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Parts : %d\n", vault.PartsSize) + + fmt.Println("=====================> Exploring vault.Other") + vault.OtherSize = 0 + vault.OtherCount = 0 + for _, item := range strings.Split(strings.TrimSuffix(vault.Other, "\n"), "\n") { +// fmt.Printf("%s : %#v\n", item, r.FindStringSubmatch(item)) +// fmt.Printf("%s : %#v\n", item, r.SubexpNames()) + if r.FindStringSubmatch(item) != nil { + if _, ok := items[r.ReplaceAllString(item, "${Code}")]; ok { + if items[r.ReplaceAllString(item, "${Code}")].Weight == -1 { + fmt.Printf("=====================> Unknown size for item : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + } else { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.OtherSize = vault.OtherSize + items[r.ReplaceAllString(item, "${Code}")].Weight * int32(size) + vault.OtherCount = vault.OtherCount + int32(size) + } + } else { + fmt.Printf("=====================> Item not found in map : %s (%s)\n", r.ReplaceAllString(item, "${Code}"), r.ReplaceAllString(item, "${Name}")) + itemName := r.ReplaceAllString(item, "${Name}") + if s.FindStringSubmatch(itemName) != nil { + for k, v := range items { + if strings.Compare(v.Name, s.ReplaceAllString(itemName, "${BaseName}")) == 0 { + size, _ = strconv.Atoi(r.ReplaceAllString(item, "${Quantity}")) + vault.OtherSize = vault.OtherSize + items[k].Weight * int32(size) + vault.OtherCount = vault.OtherCount + int32(size) + fmt.Printf("=====================> Found in map : %s (%d)\n", items[k].Name, size) + } + } + } + } + } else { + if item != "Guild Warehouse:" { + fmt.Printf("=====================> Item not matched : %s\n", item) + } + } + } + fmt.Printf("=====================> Size for vault.Other : %d\n", vault.OtherSize) + + msgText = fmt.Sprintf("Current stock [%d/38000] :\n - Resources : %d (%d)\n - Alchemist : %d (%d)\n - Misc stuff : %d (%d)\n - Recipes : %d (%d)\n - Parts : %d (%d)\n - Other : %d (%d)\n", vault.ResSize + vault.AlchSize + vault.MiscSize + vault.RecSize + vault.PartsSize + vault.OtherSize, vault.ResSize, vault.ResCount, vault.AlchSize, vault.AlchCount, vault.MiscSize, vault.MiscCount, vault.RecSize, vault.RecCount, vault.PartsSize, vault.PartsCount, vault.OtherSize, vault.OtherCount) + inputMsgTxt = tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + sendMsg, _ = c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + + break + } + validID = regexp.MustCompile(`^!g_withdraw ([a-z0-9]+) ([0-9]+)$`) + if validID.MatchString(line) { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! GUILD WITHDRAWAL SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + + t := regexp.MustCompile(`!g_withdraw (?P[a-z0-9]+) (?P[0-9]+)`) + + if t.FindStringSubmatch(line) != nil { + size, _ = strconv.Atoi(t.ReplaceAllString(line, "${Quantity}")) + fmt.Printf("Withdrawing %s / %d\n", t.ReplaceAllString(line, "${Code}"), size) + m := ChatWarsWithdraw{ + Item : t.ReplaceAllString(line, "${Code}"), + Quantity : int64(size), + UserID64 : int64(senderUserID), + Time : time.Now(), + } + + b, _ := json.Marshal(m) + + md5Bytes := md5.Sum(b) + + md5String := TgEncoding.EncodeToString(md5Bytes[:]) + + withdrawals[md5String] = m; + + msgText = fmt.Sprintf("To validate @%s withdrawal of %d x %s\nClick /withdraw_%s\n", user.Username, size, items[t.ReplaceAllString(line, "${Code}")].Name, md5String) + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + + } else { + fmt.Printf("Can't match withdrawing request.\n") + } + break + } + validID = regexp.MustCompile(`^/withdraw_([ABCDEFGHIJKLMNOPQRSTUVWXYZ234567]{26})$`) + if validID.MatchString(line) { + fmt.Println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! GUILD WITHDRAW CONF SPOTTED !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!") + + u := regexp.MustCompile(`^/withdraw_(?P[ABCDEFGHIJKLMNOPQRSTUVWXYZ234567]{26})$`) + + if u.FindStringSubmatch(line) != nil { + fmt.Printf("Withdrawal %s identified\n", u.ReplaceAllString(line, "${Reference}")) + ref := u.ReplaceAllString(line, "${Reference}") + if _, ok := withdrawals[ref]; ok { + fmt.Printf("Withdrawal %s found\n", ref) + t := time.Now() + if t.Sub(withdrawals[ref].Time) > 300 * time.Second { + msgText = fmt.Sprintf("Reference %s is expired.\n", ref) + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + } else { + if withdrawals[ref].UserID64 != ownUserID64 && withdrawals[ref].UserID64 == int64(senderUserID) { + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText("You cannot validate your own withdrawal requests.\n", nil), true, true) + c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + } else { + msgText = fmt.Sprintf("/g_withdraw %s %d\n", withdrawals[ref].Item, withdrawals[ref].Quantity) + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + msgMutex.Lock() + sendMsg, _ := c.SendMessage(user_chtwrsbot, 0, false, false, nil, inputMsgTxt) + msgID64 = sendMsg.ID + msgText = string("/g_withdraw") + msgUserID64 = withdrawals[ref].UserID64 + fmt.Println("send msgID64 (g_withdraw) : ", msgID64) + delete(withdrawals, ref) + } + } + } else { + msgText = fmt.Sprintf("Reference %s is not found.\n", ref) + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + } + } else { + msgText = fmt.Sprintf("Reference is malformed\n") + inputMsgTxt := tdlib.NewInputMessageText(tdlib.NewFormattedText(msgText, nil), true, true) + c.SendMessage(chat_guild, 0, false, false, nil, inputMsgTxt) + msgText = "" + } + break + + } + } + } + } +} +/* +func ForwardOwnMsg(c *tdlib.Client) { + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID + senderUserID := updateMsg.Message.SenderUserID + if (senderUserID == ownUserID32 && chatID != ownUserID64) { + return true + } + return false + } + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { + fwdMsg := ForwardMsg(c, updateMsg.Message.ID, updateMsg.Message.ChatID, ownUserID64) + time.Sleep(2 * time.Second) + DeleteMsg(c, fwdMsg) + } + } +} +*/ +func Listen(c *tdlib.Client) { + eventFilter := func(msg *tdlib.TdMessage) bool { + updateMsg := (*msg).(*tdlib.UpdateNewMessage) + chatID := updateMsg.Message.ChatID +// senderUserID := updateMsg.Message.SenderUserID + if (chatID == user_chtwrsbot || chatID == user_botnestbot || chatID == user_chtwrscastlebot || chatID == chat_darkwing || chatID == chat_guild) { + return false + } + return true + } + + receiver := c.AddEventReceiver(&tdlib.UpdateNewMessage{}, eventFilter, 100) + for newMsg := range receiver.Chan { + updateMsg := (newMsg).(*tdlib.UpdateNewMessage) + senderUserID := updateMsg.Message.SenderUserID + mType := updateMsg.Message.Content.GetMessageContentEnum() + if mType == "messageText" { + username := "null" + if senderUserID != 0 { + user, err := c.GetUser(senderUserID) + if err != nil { + fmt.Println("Listen:", err.Error()) + continue + } else { + username = user.Username + } + } + t := time.Now() + txt := updateMsg.Message.Content.(*tdlib.MessageText).Text.Text + fmt.Printf("[%d-%02d-%02d %02d:%02d:%02d-00:00]", t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), t.Second()) + fmt.Println(" === SENT (", username, ") =====================================================================") + fmt.Println(txt, "\n") + fmt.Println("************ DETAILS ************") + fmt.Println("ID : ", updateMsg.Message.ID) + fmt.Println("SenderUserID : ", updateMsg.Message.SenderUserID) + fmt.Println("ChatID : ", updateMsg.Message.ChatID) + fmt.Println("SendingState : ", updateMsg.Message.SendingState) + fmt.Println("IsOutgoing : ", updateMsg.Message.IsOutgoing) + fmt.Println("CanBeEdited : ", updateMsg.Message.CanBeEdited) + fmt.Println("CanBeForwarded : ", updateMsg.Message.CanBeForwarded) + fmt.Println("IsChannelPost : ", updateMsg.Message.IsChannelPost) + fmt.Println("ContainsUnreadMention : ", updateMsg.Message.ContainsUnreadMention) + fmt.Println("ForwardInfo : ", updateMsg.Message.ForwardInfo) + fmt.Println("ReplyToMessageID : ", updateMsg.Message.ReplyToMessageID) + fmt.Println("ViaBotUserID : ", updateMsg.Message.ViaBotUserID) + fmt.Println("================================================================================================================") + } + } +} + +func ForwardMsg(c *tdlib.Client, msgID int64, fromChatID int64, toChatID int64) (int64) { + msgIDs := make([]int64, 1) + msgIDs[0] = msgID + msgs, _ := c.ForwardMessages(toChatID, fromChatID, msgIDs, false, false, false) + if msgs != nil { + return msgs.Messages[0].ID + } else { + return 0 + } +} + +func DeleteMsg(c *tdlib.Client, chatID int64, msgID int64) { + msgIDs := make([]int64, 1) + msgIDs[0] = msgID + c.DeleteMessages(chatID, msgIDs, false) + fmt.Println("Deleting message ", msgID) +} + +func OwnUserID(c *tdlib.Client) (int32) { + user, _ := c.GetMe() + return user.ID; +} +