mirror of https://github.com/statping/statping
				
				
				
			
		
			
				
	
	
		
			480 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			480 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
// Statping
 | 
						|
// Copyright (C) 2018.  Hunter Long and the project contributors
 | 
						|
// Written by Hunter Long <info@socialeck.com> and the project contributors
 | 
						|
//
 | 
						|
// https://github.com/hunterlong/statping
 | 
						|
//
 | 
						|
// The licenses for most software and other practical works are designed
 | 
						|
// to take away your freedom to share and change the works.  By contrast,
 | 
						|
// the GNU General Public License is intended to guarantee your freedom to
 | 
						|
// share and change all versions of a program--to make sure it remains free
 | 
						|
// software for all its users.
 | 
						|
//
 | 
						|
// You should have received a copy of the GNU General Public License
 | 
						|
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
package notifier
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/json"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"github.com/hunterlong/statping/types"
 | 
						|
	"github.com/hunterlong/statping/utils"
 | 
						|
	"github.com/jinzhu/gorm"
 | 
						|
	"reflect"
 | 
						|
	"strings"
 | 
						|
	"time"
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	// AllCommunications holds all the loaded notifiers
 | 
						|
	AllCommunications []types.AllNotifiers
 | 
						|
	// db holds the Statping database connection
 | 
						|
	db       *gorm.DB
 | 
						|
	timezone float32
 | 
						|
)
 | 
						|
 | 
						|
// Notification contains all the fields for a Statping Notifier.
 | 
						|
type Notification struct {
 | 
						|
	Id          int64              `gorm:"primary_key;column:id" json:"id"`
 | 
						|
	Method      string             `gorm:"column:method" json:"method"`
 | 
						|
	Host        string             `gorm:"not null;column:host" json:"host,omitempty"`
 | 
						|
	Port        int                `gorm:"not null;column:port" json:"port,omitempty"`
 | 
						|
	Username    string             `gorm:"not null;column:username" json:"username,omitempty"`
 | 
						|
	Password    string             `gorm:"not null;column:password" json:"password,omitempty"`
 | 
						|
	Var1        string             `gorm:"not null;column:var1" json:"var1,omitempty"`
 | 
						|
	Var2        string             `gorm:"not null;column:var2" json:"var2,omitempty"`
 | 
						|
	ApiKey      string             `gorm:"not null;column:api_key" json:"api_key,omitempty"`
 | 
						|
	ApiSecret   string             `gorm:"not null;column:api_secret" json:"api_secret,omitempty"`
 | 
						|
	Enabled     types.NullBool     `gorm:"column:enabled;type:boolean;default:false" json:"enabled"`
 | 
						|
	Limits      int                `gorm:"not null;column:limits" json:"limits"`
 | 
						|
	Removable   bool               `gorm:"column:removable" json:"removeable"`
 | 
						|
	CreatedAt   time.Time          `gorm:"column:created_at" json:"created_at"`
 | 
						|
	UpdatedAt   time.Time          `gorm:"column:updated_at" json:"updated_at"`
 | 
						|
	Form        []NotificationForm `gorm:"-" json:"form"`
 | 
						|
	logs        []*NotificationLog `gorm:"-" json:"logs"`
 | 
						|
	Title       string             `gorm:"-" json:"title"`
 | 
						|
	Description string             `gorm:"-" json:"description"`
 | 
						|
	Author      string             `gorm:"-" json:"author"`
 | 
						|
	AuthorUrl   string             `gorm:"-" json:"author_url"`
 | 
						|
	Icon        string             `gorm:"-" json:"icon"`
 | 
						|
	Delay       time.Duration      `gorm:"-" json:"delay,string"`
 | 
						|
	Queue       []*QueueData       `gorm:"-" json:"-"`
 | 
						|
	Running     chan bool          `gorm:"-" json:"-"`
 | 
						|
	testable    bool               `gorm:"-" json:"testable"`
 | 
						|
}
 | 
						|
 | 
						|
// QueueData is the struct for the messaging queue with service
 | 
						|
type QueueData struct {
 | 
						|
	Id   string
 | 
						|
	Data interface{}
 | 
						|
}
 | 
						|
 | 
						|
// NotificationForm contains the HTML fields for each variable/input you want the notifier to accept.
 | 
						|
type NotificationForm struct {
 | 
						|
	Type        string `json:"type"`        // the html input type (text, password, email)
 | 
						|
	Title       string `json:"title"`       // include a title for ease of use
 | 
						|
	Placeholder string `json:"placeholder"` // add a placeholder for the input
 | 
						|
	DbField     string `json:"field"`       // true variable key for input
 | 
						|
	SmallText   string `json:"small_text"`  // insert small text under a html input
 | 
						|
	Required    bool   `json:"required"`    // require this input on the html form
 | 
						|
	IsHidden    bool   `json:"hidden"`      // hide this form element from end user
 | 
						|
	IsList      bool   `json:"list"`        // make this form element a comma separated list
 | 
						|
	IsSwitch    bool   `json:"switch"`      // make the notifier a boolean true/false switch
 | 
						|
}
 | 
						|
 | 
						|
// NotificationLog contains the normalized message from previously sent notifications
 | 
						|
type NotificationLog struct {
 | 
						|
	Message   string          `json:"message"`
 | 
						|
	Time      utils.Timestamp `json:"time"`
 | 
						|
	Timestamp time.Time       `json:"timestamp"`
 | 
						|
}
 | 
						|
 | 
						|
// AfterFind for Notification will set the timezone
 | 
						|
func (n *Notification) AfterFind() (err error) {
 | 
						|
	n.CreatedAt = utils.Timezoner(n.CreatedAt, timezone)
 | 
						|
	n.UpdatedAt = utils.Timezoner(n.UpdatedAt, timezone)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// AddQueue will add any type of interface (json, string, struct, etc) into the Notifiers queue
 | 
						|
func (n *Notification) AddQueue(uid string, msg interface{}) {
 | 
						|
	data := &QueueData{uid, msg}
 | 
						|
	n.Queue = append(n.Queue, data)
 | 
						|
	utils.Log(1, fmt.Sprintf("Notifier '%v' added new item (%v) to the queue. (%v queued)", n.Method, uid, len(n.Queue)))
 | 
						|
}
 | 
						|
 | 
						|
// CanTest returns true if the notifier implements the OnTest interface
 | 
						|
func (n *Notification) CanTest() bool {
 | 
						|
	return n.testable
 | 
						|
}
 | 
						|
 | 
						|
// db will return the notifier database column/record
 | 
						|
func modelDb(n *Notification) *gorm.DB {
 | 
						|
	return db.Model(&Notification{}).Where("method = ?", n.Method).Find(n)
 | 
						|
}
 | 
						|
 | 
						|
// SetDB is called by core to inject the database for a notifier to use
 | 
						|
func SetDB(d *gorm.DB, zone float32) {
 | 
						|
	db = d
 | 
						|
	timezone = zone
 | 
						|
}
 | 
						|
 | 
						|
// asNotification accepts a Notifier and returns a Notification struct
 | 
						|
func asNotification(n Notifier) *Notification {
 | 
						|
	return n.Select()
 | 
						|
}
 | 
						|
 | 
						|
// AddNotifier accept a Notifier interface to be added into the array
 | 
						|
func AddNotifiers(notifiers ...Notifier) error {
 | 
						|
	for _, n := range notifiers {
 | 
						|
		if isType(n, new(Notifier)) {
 | 
						|
			err := checkNotifierForm(n)
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
			AllCommunications = append(AllCommunications, n)
 | 
						|
			Init(n)
 | 
						|
		} else {
 | 
						|
			return errors.New("notifier does not have the required methods")
 | 
						|
		}
 | 
						|
	}
 | 
						|
	startAllNotifiers()
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// normalizeType will accept multiple interfaces and converts it into a string for logging
 | 
						|
func normalizeType(ty interface{}) string {
 | 
						|
	switch v := ty.(type) {
 | 
						|
	case int, int32, int64:
 | 
						|
		return fmt.Sprintf("%v", v)
 | 
						|
	case float32, float64:
 | 
						|
		return fmt.Sprintf("%v", v)
 | 
						|
	case string:
 | 
						|
		return v
 | 
						|
	case []byte:
 | 
						|
		return string(v)
 | 
						|
	case []string:
 | 
						|
		return fmt.Sprintf("%v", v)
 | 
						|
	case interface{}, map[string]interface{}:
 | 
						|
		j, _ := json.Marshal(v)
 | 
						|
		return string(j)
 | 
						|
	default:
 | 
						|
		return fmt.Sprintf("%v", v)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Log will record a new notification into memory and will show the logs on the settings page
 | 
						|
func (n *Notification) makeLog(msg interface{}) {
 | 
						|
	log := &NotificationLog{
 | 
						|
		Message:   normalizeType(msg),
 | 
						|
		Time:      utils.Timestamp(time.Now()),
 | 
						|
		Timestamp: time.Now(),
 | 
						|
	}
 | 
						|
	n.logs = append(n.logs, log)
 | 
						|
}
 | 
						|
 | 
						|
// Logs returns an array of the notifiers logs
 | 
						|
func (n *Notification) Logs() []*NotificationLog {
 | 
						|
	return reverseLogs(n.logs)
 | 
						|
}
 | 
						|
 | 
						|
// reverseLogs will reverse the notifier's logs to be time desc
 | 
						|
func reverseLogs(input []*NotificationLog) []*NotificationLog {
 | 
						|
	if len(input) == 0 {
 | 
						|
		return input
 | 
						|
	}
 | 
						|
	return append(reverseLogs(input[1:]), input[0])
 | 
						|
}
 | 
						|
 | 
						|
// isInDatabase returns true if the notifier has already been installed
 | 
						|
func isInDatabase(n Notifier) bool {
 | 
						|
	inDb := modelDb(n.Select()).RecordNotFound()
 | 
						|
	return !inDb
 | 
						|
}
 | 
						|
 | 
						|
// SelectNotification returns the Notification struct from the database
 | 
						|
func SelectNotification(n Notifier) (*Notification, error) {
 | 
						|
	notifier := n.Select()
 | 
						|
	err := db.Model(&Notification{}).Where("method = ?", notifier.Method).Scan(¬ifier)
 | 
						|
	return notifier, err.Error
 | 
						|
}
 | 
						|
 | 
						|
// Update will update the notification into the database
 | 
						|
func Update(n Notifier, notif *Notification) (*Notification, error) {
 | 
						|
	notif.ResetQueue()
 | 
						|
	err := db.Model(&Notification{}).Update(notif)
 | 
						|
	if notif.Enabled.Bool {
 | 
						|
		notif.close()
 | 
						|
		notif.start()
 | 
						|
		go Queue(n)
 | 
						|
	} else {
 | 
						|
		notif.close()
 | 
						|
	}
 | 
						|
	return notif, err.Error
 | 
						|
}
 | 
						|
 | 
						|
// insertDatabase will create a new record into the database for the notifier
 | 
						|
func insertDatabase(n Notifier) (int64, error) {
 | 
						|
	noti := n.Select()
 | 
						|
	noti.Limits = 3
 | 
						|
	query := db.Create(noti)
 | 
						|
	if query.Error != nil {
 | 
						|
		return 0, query.Error
 | 
						|
	}
 | 
						|
	return noti.Id, query.Error
 | 
						|
}
 | 
						|
 | 
						|
// SelectNotifier returns the Notification struct from the database
 | 
						|
func SelectNotifier(method string) (*Notification, Notifier, error) {
 | 
						|
	for _, comm := range AllCommunications {
 | 
						|
		n, ok := comm.(Notifier)
 | 
						|
		if !ok {
 | 
						|
			return nil, nil, fmt.Errorf("incorrect notification type: %v", reflect.TypeOf(n).String())
 | 
						|
		}
 | 
						|
		notifier := n.Select()
 | 
						|
		if notifier.Method == method {
 | 
						|
			return notifier, comm.(Notifier), nil
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil, nil, errors.New("cannot find notifier")
 | 
						|
}
 | 
						|
 | 
						|
// Init accepts the Notifier interface to initialize the notifier
 | 
						|
func Init(n Notifier) (*Notification, error) {
 | 
						|
	err := install(n)
 | 
						|
	var notify *Notification
 | 
						|
	if err == nil {
 | 
						|
		notify, _ = SelectNotification(n)
 | 
						|
		notify.CreatedAt = utils.Timezoner(notify.CreatedAt, timezone)
 | 
						|
		notify.UpdatedAt = utils.Timezoner(notify.UpdatedAt, timezone)
 | 
						|
		if notify.Delay.Seconds() == 0 {
 | 
						|
			notify.Delay = time.Duration(1 * time.Second)
 | 
						|
		}
 | 
						|
		notify.testable = isType(n, new(Tester))
 | 
						|
		notify.Form = n.Select().Form
 | 
						|
	}
 | 
						|
	return notify, err
 | 
						|
}
 | 
						|
 | 
						|
// startAllNotifiers will start the go routine for each loaded notifier
 | 
						|
func startAllNotifiers() {
 | 
						|
	for _, comm := range AllCommunications {
 | 
						|
		if isType(comm, new(Notifier)) {
 | 
						|
			notify := comm.(Notifier)
 | 
						|
			if notify.Select().Enabled.Bool {
 | 
						|
				notify.Select().close()
 | 
						|
				notify.Select().start()
 | 
						|
				go Queue(notify)
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Queue is the FIFO go routine to send notifications when objects are triggered
 | 
						|
func Queue(n Notifier) {
 | 
						|
	notification := n.Select()
 | 
						|
	rateLimit := notification.Delay
 | 
						|
 | 
						|
CheckNotifier:
 | 
						|
	for {
 | 
						|
		select {
 | 
						|
		case <-notification.Running:
 | 
						|
			break CheckNotifier
 | 
						|
		case <-time.After(rateLimit):
 | 
						|
			notification = n.Select()
 | 
						|
			if len(notification.Queue) > 0 {
 | 
						|
				ok, _ := notification.WithinLimits()
 | 
						|
				if ok {
 | 
						|
					msg := notification.Queue[0]
 | 
						|
					err := n.Send(msg.Data)
 | 
						|
					if err != nil {
 | 
						|
						utils.Log(2, fmt.Sprintf("Notifier '%v' had an error: %v", notification.Method, err))
 | 
						|
					} else {
 | 
						|
						utils.Log(1, fmt.Sprintf("Notifier '%v' sent outgoing message (%v) %v left in queue.", notification.Method, msg.Id, len(notification.Queue)))
 | 
						|
					}
 | 
						|
					notification.makeLog(msg.Data)
 | 
						|
					if len(notification.Queue) > 1 {
 | 
						|
						notification.Queue = notification.Queue[1:]
 | 
						|
					} else {
 | 
						|
						notification.Queue = nil
 | 
						|
					}
 | 
						|
					rateLimit = notification.Delay
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		continue
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// install will check the database for the notification, if its not inserted it will insert a new record for it
 | 
						|
func install(n Notifier) error {
 | 
						|
	inDb := isInDatabase(n)
 | 
						|
	if !inDb {
 | 
						|
		_, err := insertDatabase(n)
 | 
						|
		if err != nil {
 | 
						|
			utils.Log(3, err)
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// LastSent returns a time.Duration of the last sent notification for the notifier
 | 
						|
func (n *Notification) LastSent() time.Duration {
 | 
						|
	if len(n.logs) == 0 {
 | 
						|
		return time.Duration(0)
 | 
						|
	}
 | 
						|
	last := n.Logs()[0]
 | 
						|
	since := time.Since(last.Timestamp)
 | 
						|
	return since
 | 
						|
}
 | 
						|
 | 
						|
// SentLastHour returns the total amount of notifications sent in last 1 hour
 | 
						|
func (n *Notification) SentLastHour() int {
 | 
						|
	since := time.Now().Add(-1 * time.Hour)
 | 
						|
	return n.SentLast(since)
 | 
						|
}
 | 
						|
 | 
						|
// SentLastMinute returns the total amount of notifications sent in last 1 minute
 | 
						|
func (n *Notification) SentLastMinute() int {
 | 
						|
	since := time.Now().Add(-1 * time.Minute)
 | 
						|
	return n.SentLast(since)
 | 
						|
}
 | 
						|
 | 
						|
// SentLast accept a time.Time and returns the amount of sent notifications within your time to current
 | 
						|
func (n *Notification) SentLast(since time.Time) int {
 | 
						|
	sent := 0
 | 
						|
	for _, v := range n.Logs() {
 | 
						|
		lastTime := time.Time(v.Time)
 | 
						|
		if lastTime.After(since) {
 | 
						|
			sent++
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return sent
 | 
						|
}
 | 
						|
 | 
						|
// GetValue returns the database value of a accept DbField value.
 | 
						|
func (n *Notification) GetValue(dbField string) string {
 | 
						|
	dbField = strings.ToLower(dbField)
 | 
						|
	switch dbField {
 | 
						|
	case "host":
 | 
						|
		return n.Host
 | 
						|
	case "port":
 | 
						|
		return fmt.Sprintf("%v", n.Port)
 | 
						|
	case "username":
 | 
						|
		return n.Username
 | 
						|
	case "password":
 | 
						|
		if n.Password != "" {
 | 
						|
			return "##########"
 | 
						|
		}
 | 
						|
	case "var1":
 | 
						|
		return n.Var1
 | 
						|
	case "var2":
 | 
						|
		return n.Var2
 | 
						|
	case "api_key":
 | 
						|
		return n.ApiKey
 | 
						|
	case "api_secret":
 | 
						|
		return n.ApiSecret
 | 
						|
	case "limits":
 | 
						|
		return utils.ToString(int(n.Limits))
 | 
						|
	}
 | 
						|
	return ""
 | 
						|
}
 | 
						|
 | 
						|
// isType will return true if a variable can implement an interface
 | 
						|
func isType(n interface{}, obj interface{}) bool {
 | 
						|
	one := reflect.TypeOf(n)
 | 
						|
	two := reflect.ValueOf(obj).Elem()
 | 
						|
	return one.Implements(two.Type())
 | 
						|
}
 | 
						|
 | 
						|
// isEnabled returns true if the notifier is enabled
 | 
						|
func isEnabled(n interface{}) bool {
 | 
						|
	notifier := n.(Notifier).Select()
 | 
						|
	return notifier.Enabled.Bool
 | 
						|
}
 | 
						|
 | 
						|
// inLimits will return true if the notifier is within sending limits
 | 
						|
func inLimits(n interface{}) bool {
 | 
						|
	notifier := n.(Notifier).Select()
 | 
						|
	ok, _ := notifier.WithinLimits()
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
// WithinLimits returns true if the notifier is within its sending limits
 | 
						|
func (n *Notification) WithinLimits() (bool, error) {
 | 
						|
	if n.SentLastMinute() == 0 {
 | 
						|
		return true, nil
 | 
						|
	}
 | 
						|
	if n.SentLastMinute() >= n.Limits {
 | 
						|
		return false, fmt.Errorf("notifier sent %v out of %v in last minute", n.SentLastMinute(), n.Limits)
 | 
						|
	}
 | 
						|
	if n.LastSent().Seconds() == 0 {
 | 
						|
		return true, nil
 | 
						|
	}
 | 
						|
	if n.Delay.Seconds() >= n.LastSent().Seconds() {
 | 
						|
		return false, fmt.Errorf("notifiers delay (%v) is greater than last message sent (%v)", n.Delay.Seconds(), n.LastSent().Seconds())
 | 
						|
	}
 | 
						|
	return true, nil
 | 
						|
}
 | 
						|
 | 
						|
// ResetQueue will clear the notifiers Queue
 | 
						|
func (n *Notification) ResetQueue() {
 | 
						|
	n.Queue = nil
 | 
						|
}
 | 
						|
 | 
						|
// ResetQueue will clear the notifiers Queue for a service
 | 
						|
func (n *Notification) ResetUniqueQueue(uid string) []*QueueData {
 | 
						|
	var queue []*QueueData
 | 
						|
	for _, v := range n.Queue {
 | 
						|
		if v.Id != uid {
 | 
						|
			queue = append(queue, v)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	n.Queue = queue
 | 
						|
	return queue
 | 
						|
}
 | 
						|
 | 
						|
// start will start the go routine for the notifier queue
 | 
						|
func (n *Notification) start() {
 | 
						|
	n.Running = make(chan bool)
 | 
						|
}
 | 
						|
 | 
						|
// close will stop the go routine for queue
 | 
						|
func (n *Notification) close() {
 | 
						|
	if n.IsRunning() {
 | 
						|
		close(n.Running)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// IsRunning will return true if the notifier is currently running a queue
 | 
						|
func (n *Notification) IsRunning() bool {
 | 
						|
	if n.Running == nil {
 | 
						|
		return false
 | 
						|
	}
 | 
						|
	select {
 | 
						|
	case <-n.Running:
 | 
						|
		return false
 | 
						|
	default:
 | 
						|
		return true
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ExampleService can be used for the OnTest() method for notifiers
 | 
						|
var ExampleService = &types.Service{
 | 
						|
	Id:             1,
 | 
						|
	Name:           "Interpol - All The Rage Back Home",
 | 
						|
	Domain:         "https://www.youtube.com/watch?v=-u6DvRyyKGU",
 | 
						|
	ExpectedStatus: 200,
 | 
						|
	Interval:       30,
 | 
						|
	Type:           "http",
 | 
						|
	Method:         "GET",
 | 
						|
	Timeout:        20,
 | 
						|
	LastStatusCode: 404,
 | 
						|
	Expected:       types.NewNullString("test example"),
 | 
						|
	LastResponse:   "<html>this is an example response</html>",
 | 
						|
	CreatedAt:      time.Now().Add(-24 * time.Hour),
 | 
						|
}
 |