statping/core/checker.go

282 lines
7.3 KiB
Go
Raw Normal View History

2018-12-04 05:57:11 +00:00
// Statping
2018-08-16 06:22:20 +00:00
// Copyright (C) 2018. Hunter Long and the project contributors
// Written by Hunter Long <info@socialeck.com> and the project contributors
//
2018-12-04 04:17:29 +00:00
// https://github.com/hunterlong/statping
2018-08-16 06:22:20 +00:00
//
// 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/>.
2018-06-30 00:57:05 +00:00
package core
2018-06-10 01:31:13 +00:00
import (
"bytes"
2018-06-10 01:31:13 +00:00
"fmt"
2018-12-04 04:17:29 +00:00
"github.com/hunterlong/statping/core/notifier"
"github.com/hunterlong/statping/types"
"github.com/hunterlong/statping/utils"
2019-04-20 03:41:09 +00:00
"github.com/tatsushid/go-fastping"
"net"
2018-06-10 01:31:13 +00:00
"net/http"
"net/url"
2018-06-10 01:31:13 +00:00
"regexp"
"strings"
2018-06-10 01:31:13 +00:00
"time"
)
// checkServices will start the checking go routine for each service
func checkServices() {
utils.Log(1, fmt.Sprintf("Starting monitoring process for %v Services", len(CoreApp.Services)))
for _, ser := range CoreApp.Services {
2018-07-02 06:21:41 +00:00
//go obj.StartCheckins()
2018-08-21 01:22:42 +00:00
go ser.CheckQueue(true)
2018-06-10 01:31:13 +00:00
}
}
2019-04-20 03:41:09 +00:00
// Check will run checkHttp for HTTP services and checkTcp for TCP services
// if record param is set to true, it will add a record into the database.
2019-04-20 03:41:09 +00:00
func (s *Service) Check(record bool) {
switch s.Type {
case "http":
s.checkHttp(record)
case "tcp", "udp":
s.checkTcp(record)
case "icmp":
s.checkIcmp(record)
}
}
// CheckQueue is the main go routine for checking a service
func (s *Service) CheckQueue(record bool) {
2018-08-20 07:59:17 +00:00
s.Checkpoint = time.Now()
s.SleepDuration = time.Duration((time.Duration(s.Id) * 100) * time.Millisecond)
2018-08-19 00:37:00 +00:00
CheckLoop:
for {
select {
2018-08-19 00:37:00 +00:00
case <-s.Running:
utils.Log(1, fmt.Sprintf("Stopping service: %v", s.Name))
break CheckLoop
case <-time.After(s.SleepDuration):
2018-08-21 01:22:42 +00:00
s.Check(record)
2018-08-21 06:54:39 +00:00
s.Checkpoint = s.Checkpoint.Add(s.duration())
sleep := s.Checkpoint.Sub(time.Now())
if !s.Online {
s.SleepDuration = s.duration()
} else {
s.SleepDuration = sleep
}
}
continue
2018-06-15 04:30:10 +00:00
}
2018-06-10 01:31:13 +00:00
}
// duration returns the amount of duration for a service to check its status
2018-08-21 06:54:39 +00:00
func (s *Service) duration() time.Duration {
var amount time.Duration
if s.Interval >= 10000 {
amount = time.Duration(s.Interval) * time.Microsecond
} else {
amount = time.Duration(s.Interval) * time.Second
}
return amount
}
func (s *Service) parseHost() string {
2018-10-07 02:44:31 +00:00
if s.Type == "tcp" || s.Type == "udp" {
return s.Domain
} else {
2019-05-27 17:35:51 +00:00
u, err := url.Parse(s.Domain)
if err != nil {
return s.Domain
}
2019-05-27 17:35:51 +00:00
return strings.Split(u.Host, ":")[0]
}
}
// dnsCheck will check the domain name and return a float64 for the amount of time the DNS check took
2018-08-21 01:22:42 +00:00
func (s *Service) dnsCheck() (float64, error) {
var err error
t1 := time.Now()
host := s.parseHost()
if s.Type == "tcp" {
_, err = net.LookupHost(host)
} else {
_, err = net.LookupIP(host)
}
if err != nil {
return 0, err
}
t2 := time.Now()
subTime := t2.Sub(t1).Seconds()
return subTime, err
}
2019-06-06 19:10:52 +00:00
func isIPv6(address string) bool {
return strings.Count(address, ":") >= 2
}
2019-04-20 03:41:09 +00:00
// checkIcmp will send a ICMP ping packet to the service
func (s *Service) checkIcmp(record bool) *Service {
p := fastping.NewPinger()
2019-06-06 19:10:52 +00:00
resolveIP := "ip4:icmp"
if isIPv6(s.Domain) {
resolveIP = "ip6:icmp"
}
ra, err := net.ResolveIPAddr(resolveIP, s.Domain)
2019-04-20 03:41:09 +00:00
if err != nil {
recordFailure(s, fmt.Sprintf("Could not send ICMP to service %v, %v", s.Domain, err))
return s
}
p.AddIPAddr(ra)
p.OnRecv = func(addr *net.IPAddr, rtt time.Duration) {
s.Latency = rtt.Seconds()
recordSuccess(s)
}
err = p.Run()
if err != nil {
recordFailure(s, fmt.Sprintf("Issue running ICMP to service %v, %v", s.Domain, err))
return s
}
s.LastResponse = ""
return s
}
// checkTcp will check a TCP service
2018-08-21 01:22:42 +00:00
func (s *Service) checkTcp(record bool) *Service {
dnsLookup, err := s.dnsCheck()
if err != nil {
if record {
recordFailure(s, fmt.Sprintf("Could not get IP address for TCP service %v, %v", s.Domain, err))
}
return s
}
s.PingTime = dnsLookup
t1 := time.Now()
domain := fmt.Sprintf("%v", s.Domain)
if s.Port != 0 {
domain = fmt.Sprintf("%v:%v", s.Domain, s.Port)
2019-06-06 19:10:52 +00:00
if isIPv6(s.Domain) {
domain = fmt.Sprintf("[%v]:%v", s.Domain, s.Port)
}
}
2018-10-07 02:44:31 +00:00
conn, err := net.DialTimeout(s.Type, domain, time.Duration(s.Timeout)*time.Second)
if err != nil {
2018-08-19 00:37:00 +00:00
if record {
2019-06-06 19:10:52 +00:00
recordFailure(s, fmt.Sprintf("Dial Error %v", err))
2018-08-19 00:37:00 +00:00
}
return s
}
if err := conn.Close(); err != nil {
2018-08-19 00:37:00 +00:00
if record {
2019-06-06 19:10:52 +00:00
recordFailure(s, fmt.Sprintf("%v Socket Close Error %v", strings.ToUpper(s.Type), err))
2018-08-19 00:37:00 +00:00
}
return s
}
t2 := time.Now()
s.Latency = t2.Sub(t1).Seconds()
s.LastResponse = ""
2018-08-19 00:37:00 +00:00
if record {
2018-09-25 07:03:49 +00:00
recordSuccess(s)
2018-08-19 00:37:00 +00:00
}
return s
}
// checkHttp will check a HTTP service
2018-08-21 01:22:42 +00:00
func (s *Service) checkHttp(record bool) *Service {
dnsLookup, err := s.dnsCheck()
if err != nil {
2018-08-19 00:37:00 +00:00
if record {
2018-09-25 07:03:49 +00:00
recordFailure(s, fmt.Sprintf("Could not get IP address for domain %v, %v", s.Domain, err))
2018-08-19 00:37:00 +00:00
}
return s
}
s.PingTime = dnsLookup
2018-06-10 01:31:13 +00:00
t1 := time.Now()
2018-11-25 10:18:21 +00:00
timeout := time.Duration(s.Timeout) * time.Second
var content []byte
var res *http.Response
2019-03-16 22:55:45 +00:00
var headers []string
if s.Headers.Valid {
headers = strings.Split(s.Headers.String, ",")
} else {
headers = nil
}
if s.Method == "POST" {
2019-03-16 22:55:45 +00:00
content, res, err = utils.HttpRequest(s.Domain, s.Method, "application/json", headers, bytes.NewBuffer([]byte(s.PostData.String)), timeout)
} else {
2019-03-16 22:55:45 +00:00
content, res, err = utils.HttpRequest(s.Domain, s.Method, nil, headers, nil, timeout)
}
2018-06-30 22:37:01 +00:00
if err != nil {
2018-08-19 00:37:00 +00:00
if record {
2018-09-25 07:03:49 +00:00
recordFailure(s, fmt.Sprintf("HTTP Error %v", err))
2018-08-19 00:37:00 +00:00
}
2018-06-30 22:37:01 +00:00
return s
}
2018-06-10 01:31:13 +00:00
t2 := time.Now()
s.Latency = t2.Sub(t1).Seconds()
2018-11-25 10:18:21 +00:00
s.LastResponse = string(content)
s.LastStatusCode = res.StatusCode
if s.Expected.String != "" {
2018-11-25 10:18:21 +00:00
match, err := regexp.MatchString(s.Expected.String, string(content))
if err != nil {
utils.Log(2, fmt.Sprintf("Service %v expected: %v to match %v", s.Name, string(content), s.Expected.String))
}
2018-06-10 01:31:13 +00:00
if !match {
2018-08-19 00:37:00 +00:00
if record {
2018-09-25 07:03:49 +00:00
recordFailure(s, fmt.Sprintf("HTTP Response Body did not match '%v'", s.Expected))
2018-08-19 00:37:00 +00:00
}
2018-06-15 04:30:10 +00:00
return s
2018-06-10 01:31:13 +00:00
}
}
2018-11-25 10:18:21 +00:00
if s.ExpectedStatus != res.StatusCode {
2018-08-19 00:37:00 +00:00
if record {
2018-11-25 10:18:21 +00:00
recordFailure(s, fmt.Sprintf("HTTP Status Code %v did not match %v", res.StatusCode, s.ExpectedStatus))
2018-08-19 00:37:00 +00:00
}
2018-06-15 04:30:10 +00:00
return s
2018-06-10 01:31:13 +00:00
}
2018-08-19 00:37:00 +00:00
if record {
2018-09-25 07:03:49 +00:00
recordSuccess(s)
2018-08-19 00:37:00 +00:00
}
2018-06-15 04:30:10 +00:00
return s
}
2018-09-25 07:03:49 +00:00
// recordSuccess will create a new 'hit' record in the database for a successful/online service
func recordSuccess(s *Service) {
s.LastOnline = utils.Timezoner(time.Now().UTC(), CoreApp.Timezone)
hit := &types.Hit{
Service: s.Id,
Latency: s.Latency,
PingTime: s.PingTime,
CreatedAt: time.Now(),
2018-06-15 04:30:10 +00:00
}
utils.Log(1, fmt.Sprintf("Service %v Successful Response: %0.2f ms | Lookup in: %0.2f ms", s.Name, hit.Latency*1000, hit.PingTime*1000))
s.CreateHit(hit)
s.Online = true
notifier.OnSuccess(s.Service)
2018-06-10 01:31:13 +00:00
}
2018-12-06 19:03:55 +00:00
// recordFailure will create a new 'Failure' record in the database for a offline service
2018-09-25 07:03:49 +00:00
func recordFailure(s *Service, issue string) {
2018-12-06 19:03:55 +00:00
fail := &Failure{&types.Failure{
Service: s.Id,
Issue: issue,
PingTime: s.PingTime,
CreatedAt: time.Now(),
2019-01-29 12:02:13 +00:00
ErrorCode: s.LastStatusCode,
2018-11-08 10:50:06 +00:00
}}
utils.Log(2, fmt.Sprintf("Service %v Failing: %v | Lookup in: %0.2f ms", s.Name, issue, fail.PingTime*1000))
s.CreateFailure(fail)
s.Online = false
notifier.OnFailure(s.Service, fail.Failure)
2018-06-10 01:31:13 +00:00
}