package singleton import ( "crypto/md5" // #nosec "encoding/hex" "log" "regexp" "strconv" "strings" "sync" "time" "github.com/naiba/nezha/model" ) const firstNotificationDelay = time.Minute * 15 // 通知方式 var ( NotificationList map[string]map[uint64]*model.Notification // [NotificationMethodTag][NotificationID] -> model.Notification NotificationIDToTag map[uint64]string // [NotificationID] -> NotificationTag notificationsLock sync.RWMutex ) // InitNotification 初始化 Tag <-> ID <-> Notification 的映射 func InitNotification() { NotificationList = make(map[string]map[uint64]*model.Notification) NotificationIDToTag = make(map[uint64]string) } // LoadNotifications 从 DB 初始化通知方式相关参数 func LoadNotifications() { InitNotification() notificationsLock.Lock() defer notificationsLock.Unlock() var notifications []model.Notification if err := DB.Find(¬ifications).Error; err != nil { panic(err) } for i := 0; i < len(notifications); i++ { // 旧版本的Tag可能不存在 自动设置为默认值 if notifications[i].Tag == "" { SetDefaultNotificationTagInDB(¬ifications[i]) } AddNotificationToList(¬ifications[i]) } } // SetDefaultNotificationTagInDB 设置默认通知方式的 Tag func SetDefaultNotificationTagInDB(n *model.Notification) { n.Tag = "default" if err := DB.Save(n).Error; err != nil { log.Println("[ERROR]", err) } } // OnRefreshOrAddNotification 刷新通知方式相关参数 func OnRefreshOrAddNotification(n *model.Notification) { notificationsLock.Lock() defer notificationsLock.Unlock() var isEdit bool if _, ok := NotificationIDToTag[n.ID]; ok { isEdit = true } if !isEdit { AddNotificationToList(n) } else { UpdateNotificationInList(n) } } // AddNotificationToList 添加通知方式到map中 func AddNotificationToList(n *model.Notification) { // 当前 Tag 不存在,创建对应该 Tag 的 子 map 后再添加 if _, ok := NotificationList[n.Tag]; !ok { NotificationList[n.Tag] = make(map[uint64]*model.Notification) } NotificationList[n.Tag][n.ID] = n NotificationIDToTag[n.ID] = n.Tag } // UpdateNotificationInList 在 map 中更新通知方式 func UpdateNotificationInList(n *model.Notification) { if n.Tag != NotificationIDToTag[n.ID] { // 如果 Tag 不一致,则需要先移除原有的映射关系 delete(NotificationList[NotificationIDToTag[n.ID]], n.ID) delete(NotificationIDToTag, n.ID) // 将新的 Tag 中的通知方式添加到 map 中 AddNotificationToList(n) } else { // 如果 Tag 一致,则直接更新 NotificationList[n.Tag][n.ID] = n } } // OnDeleteNotification 在map中删除通知方式 func OnDeleteNotification(id uint64) { notificationsLock.Lock() defer notificationsLock.Unlock() delete(NotificationList[NotificationIDToTag[id]], id) delete(NotificationIDToTag, id) } // SendNotification 向指定的通知方式组的所有通知方式发送通知 func SendNotification(notificationTag string, desc string, mutable bool) { if mutable { // 通知防骚扰策略 nID := hex.EncodeToString(md5.New().Sum([]byte(desc))) // #nosec var flag bool if cacheN, has := Cache.Get(nID); has { nHistory := cacheN.(NotificationHistory) // 每次提醒都增加一倍等待时间,最后每天最多提醒一次 if time.Now().After(nHistory.Until) { flag = true nHistory.Duration *= 2 if nHistory.Duration > time.Hour*24 { nHistory.Duration = time.Hour * 24 } nHistory.Until = time.Now().Add(nHistory.Duration) // 缓存有效期加 10 分钟 Cache.Set(nID, nHistory, nHistory.Duration+time.Minute*10) } } else { // 新提醒直接通知 flag = true Cache.Set(nID, NotificationHistory{ Duration: firstNotificationDelay, Until: time.Now().Add(firstNotificationDelay), }, firstNotificationDelay+time.Minute*10) } if !flag { if Conf.Debug { log.Println("NEZHA>> 静音的重复通知:", desc, mutable) } return } } // 向该通知方式组的所有通知方式发出通知 notificationsLock.RLock() defer notificationsLock.RUnlock() for _, n := range NotificationList[notificationTag] { log.Println("尝试通知", n.Name) } for _, n := range NotificationList[notificationTag] { desc, server := findServerInMsg(desc) ns := model.NotificationServerBundle{ Notification: n, Server: server, } if err := ns.Send(desc); err != nil { log.Println("NEZHA>> 向 ", n.Name, " 发送通知失败:", err) } else { log.Println("NEZHA>> 向 ", n.Name, " 发送通知成功:") } } } // findServerInMsg 通过msg字符串中的$ServerID$ 返回修改后的字符串与Server对象 func findServerInMsg(msg string) (string, *model.Server) { reg1 := regexp.MustCompile(`^\$\d+`) reg2 := regexp.MustCompile(`[^$]+`) ServerIDStr := reg2.FindString(reg1.FindString(msg)) ServerID, _ := strconv.ParseUint(ServerIDStr, 10, 64) // 将原字符串的ServerID标识去除 if ServerIDStr != "" { msg = strings.Replace(msg, "$"+ServerIDStr+"$", "", 1) } return msg, ServerList[ServerID] }