mirror of https://github.com/statping/statping
				
				
				
			
		
			
				
	
	
		
			268 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			7.3 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 core
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"github.com/ararog/timeago"
 | 
						|
	"github.com/hunterlong/statping/types"
 | 
						|
	"github.com/hunterlong/statping/utils"
 | 
						|
	"sort"
 | 
						|
	"time"
 | 
						|
)
 | 
						|
 | 
						|
type Checkin struct {
 | 
						|
	*types.Checkin
 | 
						|
}
 | 
						|
 | 
						|
type CheckinHit struct {
 | 
						|
	*types.CheckinHit
 | 
						|
}
 | 
						|
 | 
						|
// Select returns a *types.Checkin
 | 
						|
func (c *Checkin) Select() *types.Checkin {
 | 
						|
	return c.Checkin
 | 
						|
}
 | 
						|
 | 
						|
// Routine for checking if the last Checkin was within its interval
 | 
						|
func (c *Checkin) Routine() {
 | 
						|
	if c.Last() == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	reCheck := c.Period()
 | 
						|
CheckinLoop:
 | 
						|
	for {
 | 
						|
		select {
 | 
						|
		case <-c.Running:
 | 
						|
			utils.Log(1, fmt.Sprintf("Stopping checkin routine: %v", c.Name))
 | 
						|
			c.Failing = false
 | 
						|
			break CheckinLoop
 | 
						|
		case <-time.After(reCheck):
 | 
						|
			utils.Log(1, fmt.Sprintf("Checkin %v is expected at %v, checking every %v", c.Name, utils.FormatDuration(c.Expected()), utils.FormatDuration(c.Period())))
 | 
						|
			if c.Expected().Seconds() <= 0 {
 | 
						|
				issue := fmt.Sprintf("Checkin %v is failing, no request since %v", c.Name, c.Last().CreatedAt)
 | 
						|
				utils.Log(3, issue)
 | 
						|
				c.CreateFailure()
 | 
						|
			}
 | 
						|
			reCheck = c.Period()
 | 
						|
		}
 | 
						|
		continue
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// String will return a Checkin API string
 | 
						|
func (c *Checkin) String() string {
 | 
						|
	return c.ApiKey
 | 
						|
}
 | 
						|
 | 
						|
// ReturnCheckin converts *types.Checking to *core.Checkin
 | 
						|
func ReturnCheckin(c *types.Checkin) *Checkin {
 | 
						|
	return &Checkin{Checkin: c}
 | 
						|
}
 | 
						|
 | 
						|
// ReturnCheckinHit converts *types.checkinHit to *core.checkinHit
 | 
						|
func ReturnCheckinHit(c *types.CheckinHit) *CheckinHit {
 | 
						|
	return &CheckinHit{CheckinHit: c}
 | 
						|
}
 | 
						|
 | 
						|
func (c *Checkin) Service() *Service {
 | 
						|
	return SelectService(c.ServiceId)
 | 
						|
}
 | 
						|
 | 
						|
func (c *Checkin) CreateFailure() (int64, error) {
 | 
						|
	service := c.Service()
 | 
						|
	c.Failing = true
 | 
						|
	fail := &Failure{&types.Failure{
 | 
						|
		Issue:    fmt.Sprintf("Checkin %v was not reported %v ago, it expects a request every %v", c.Name, utils.FormatDuration(c.Expected()), utils.FormatDuration(c.Period())),
 | 
						|
		Method:   "checkin",
 | 
						|
		MethodId: c.Id,
 | 
						|
		Service:  service.Id,
 | 
						|
		Checkin:  c.Id,
 | 
						|
		PingTime: c.Expected().Seconds(),
 | 
						|
	}}
 | 
						|
	row := failuresDB().Create(&fail)
 | 
						|
	sort.Sort(types.FailSort(c.Failures))
 | 
						|
	c.Failures = append(c.Failures, fail)
 | 
						|
	if len(c.Failures) > limitedFailures {
 | 
						|
		c.Failures = c.Failures[1:]
 | 
						|
	}
 | 
						|
	return fail.Id, row.Error
 | 
						|
}
 | 
						|
 | 
						|
// LimitedHits will return the last amount of successful hits from a checkin
 | 
						|
func (c *Checkin) LimitedHits(amount int64) []*types.CheckinHit {
 | 
						|
	var hits []*types.CheckinHit
 | 
						|
	checkinHitsDB().Where("checkin = ?", c.Id).Order("id desc").Limit(amount).Find(&hits)
 | 
						|
	return hits
 | 
						|
}
 | 
						|
 | 
						|
// AllCheckins returns all checkin in system
 | 
						|
func AllCheckins() []*Checkin {
 | 
						|
	var checkins []*Checkin
 | 
						|
	checkinDB().Find(&checkins)
 | 
						|
	return checkins
 | 
						|
}
 | 
						|
 | 
						|
// SelectCheckin will find a Checkin based on the API supplied
 | 
						|
func SelectCheckin(api string) *Checkin {
 | 
						|
	for _, s := range Services() {
 | 
						|
		for _, c := range s.Select().Checkins {
 | 
						|
			if c.Select().ApiKey == api {
 | 
						|
				return c.(*Checkin)
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// Period will return the duration of the Checkin interval
 | 
						|
func (c *Checkin) Period() time.Duration {
 | 
						|
	duration, _ := time.ParseDuration(fmt.Sprintf("%vs", c.Interval))
 | 
						|
	return duration
 | 
						|
}
 | 
						|
 | 
						|
// Grace will return the duration of the Checkin Grace Period (after service hasn't responded, wait a bit for a response)
 | 
						|
func (c *Checkin) Grace() time.Duration {
 | 
						|
	duration, _ := time.ParseDuration(fmt.Sprintf("%vs", c.GracePeriod))
 | 
						|
	return duration
 | 
						|
}
 | 
						|
 | 
						|
// Expected returns the duration of when the serviec should receive a Checkin
 | 
						|
func (c *Checkin) Expected() time.Duration {
 | 
						|
	last := c.Last().CreatedAt
 | 
						|
	now := time.Now()
 | 
						|
	lastDir := now.Sub(last)
 | 
						|
	sub := time.Duration(c.Period() - lastDir)
 | 
						|
	return sub
 | 
						|
}
 | 
						|
 | 
						|
// Last returns the last checkinHit for a Checkin
 | 
						|
func (c *Checkin) Last() *CheckinHit {
 | 
						|
	var hit CheckinHit
 | 
						|
	checkinHitsDB().Where("checkin = ?", c.Id).Last(&hit)
 | 
						|
	return &hit
 | 
						|
}
 | 
						|
 | 
						|
func (c *Checkin) Link() string {
 | 
						|
	return fmt.Sprintf("%v/checkin/%v", CoreApp.Domain, c.ApiKey)
 | 
						|
}
 | 
						|
 | 
						|
// AllHits returns all of the CheckinHits for a given Checkin
 | 
						|
func (c *Checkin) AllHits() []*types.CheckinHit {
 | 
						|
	var checkins []*types.CheckinHit
 | 
						|
	checkinHitsDB().Where("checkin = ?", c.Id).Order("id DESC").Find(&checkins)
 | 
						|
	return checkins
 | 
						|
}
 | 
						|
 | 
						|
// Hits returns all of the CheckinHits for a given Checkin
 | 
						|
func (c *Checkin) LimitedFailures(amount int64) []types.FailureInterface {
 | 
						|
	var failures []*Failure
 | 
						|
	var failInterfaces []types.FailureInterface
 | 
						|
	col := failuresDB().Where("checkin = ?", c.Id).Where("method = 'checkin'").Limit(amount).Order("id desc")
 | 
						|
	col.Find(&failures)
 | 
						|
	for _, f := range failures {
 | 
						|
		failInterfaces = append(failInterfaces, f)
 | 
						|
	}
 | 
						|
	return failInterfaces
 | 
						|
}
 | 
						|
 | 
						|
// Hits returns all of the CheckinHits for a given Checkin
 | 
						|
func (c *Checkin) AllFailures() []*types.Failure {
 | 
						|
	var failures []*types.Failure
 | 
						|
	col := failuresDB().Where("checkin = ?", c.Id).Where("method = 'checkin'").Order("id desc")
 | 
						|
	col.Find(&failures)
 | 
						|
	return failures
 | 
						|
}
 | 
						|
 | 
						|
// Create will create a new Checkin
 | 
						|
func (c *Checkin) Delete() error {
 | 
						|
	c.Close()
 | 
						|
	i := c.index()
 | 
						|
	service := c.Service()
 | 
						|
	slice := service.Checkins
 | 
						|
	service.Checkins = append(slice[:i], slice[i+1:]...)
 | 
						|
	row := checkinDB().Delete(&c)
 | 
						|
	return row.Error
 | 
						|
}
 | 
						|
 | 
						|
// index returns a checkin index int for updating the *checkin.Service slice
 | 
						|
func (c *Checkin) index() int {
 | 
						|
	for k, checkin := range c.Service().Checkins {
 | 
						|
		if c.Id == checkin.Select().Id {
 | 
						|
			return k
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0
 | 
						|
}
 | 
						|
 | 
						|
// Create will create a new Checkin
 | 
						|
func (c *Checkin) Create() (int64, error) {
 | 
						|
	c.ApiKey = utils.RandomString(7)
 | 
						|
	row := checkinDB().Create(&c)
 | 
						|
	if row.Error != nil {
 | 
						|
		utils.Log(2, row.Error)
 | 
						|
		return 0, row.Error
 | 
						|
	}
 | 
						|
	service := SelectService(c.ServiceId)
 | 
						|
	service.Checkins = append(service.Checkins, c)
 | 
						|
	c.Start()
 | 
						|
	go c.Routine()
 | 
						|
	return c.Id, row.Error
 | 
						|
}
 | 
						|
 | 
						|
// Update will update a Checkin
 | 
						|
func (c *Checkin) Update() (int64, error) {
 | 
						|
	row := checkinDB().Update(&c)
 | 
						|
	if row.Error != nil {
 | 
						|
		utils.Log(2, row.Error)
 | 
						|
		return 0, row.Error
 | 
						|
	}
 | 
						|
	return c.Id, row.Error
 | 
						|
}
 | 
						|
 | 
						|
// Create will create a new successful checkinHit
 | 
						|
func (c *CheckinHit) Create() (int64, error) {
 | 
						|
	if c.CreatedAt.IsZero() {
 | 
						|
		c.CreatedAt = time.Now()
 | 
						|
	}
 | 
						|
	row := checkinHitsDB().Create(&c)
 | 
						|
	if row.Error != nil {
 | 
						|
		utils.Log(2, row.Error)
 | 
						|
		return 0, row.Error
 | 
						|
	}
 | 
						|
	return c.Id, row.Error
 | 
						|
}
 | 
						|
 | 
						|
// Ago returns the duration of time between now and the last successful checkinHit
 | 
						|
func (c *CheckinHit) Ago() string {
 | 
						|
	got, _ := timeago.TimeAgoWithTime(time.Now(), c.CreatedAt)
 | 
						|
	return got
 | 
						|
}
 | 
						|
 | 
						|
// RecheckCheckinFailure will check if a Service Checkin has been reported yet
 | 
						|
func (c *Checkin) RecheckCheckinFailure(guard chan struct{}) {
 | 
						|
	between := time.Now().Sub(time.Now()).Seconds()
 | 
						|
	if between > float64(c.Interval) {
 | 
						|
		fmt.Println("rechecking every 15 seconds!")
 | 
						|
		time.Sleep(15 * time.Second)
 | 
						|
		guard <- struct{}{}
 | 
						|
		c.RecheckCheckinFailure(guard)
 | 
						|
	} else {
 | 
						|
		fmt.Println("i recovered!!")
 | 
						|
	}
 | 
						|
	<-guard
 | 
						|
}
 |