mirror of https://github.com/statping/statping
http webhook notifier - checkins
parent
baf6ca3d34
commit
ba85a96ab8
|
@ -291,7 +291,7 @@ func (db *DbConfig) DropDatabase() error {
|
||||||
func (db *DbConfig) CreateDatabase() error {
|
func (db *DbConfig) CreateDatabase() error {
|
||||||
utils.Log(1, "Creating Database Tables...")
|
utils.Log(1, "Creating Database Tables...")
|
||||||
err := DbSession.CreateTable(&types.Checkin{})
|
err := DbSession.CreateTable(&types.Checkin{})
|
||||||
//err = DbSession.CreateTable(&types.CheckinHit{})
|
err = DbSession.CreateTable(&types.CheckinHit{})
|
||||||
err = DbSession.CreateTable(¬ifier.Notification{})
|
err = DbSession.CreateTable(¬ifier.Notification{})
|
||||||
err = DbSession.Table("core").CreateTable(&types.Core{})
|
err = DbSession.Table("core").CreateTable(&types.Core{})
|
||||||
err = DbSession.CreateTable(&types.Failure{})
|
err = DbSession.CreateTable(&types.Failure{})
|
||||||
|
@ -317,7 +317,7 @@ func (db *DbConfig) MigrateDatabase() error {
|
||||||
if tx.Error != nil {
|
if tx.Error != nil {
|
||||||
return tx.Error
|
return tx.Error
|
||||||
}
|
}
|
||||||
tx = tx.AutoMigrate(&types.Service{}, &types.User{}, &types.Hit{}, &types.Failure{}, &types.Checkin{}, ¬ifier.Notification{}).Table("core").AutoMigrate(&types.Core{})
|
tx = tx.AutoMigrate(&types.Service{}, &types.User{}, &types.Hit{}, &types.Failure{}, &types.Checkin{}, &types.CheckinHit{}, ¬ifier.Notification{}).Table("core").AutoMigrate(&types.Core{})
|
||||||
if tx.Error != nil {
|
if tx.Error != nil {
|
||||||
tx.Rollback()
|
tx.Rollback()
|
||||||
utils.Log(3, fmt.Sprintf("Statup Database could not be migrated: %v", tx.Error))
|
utils.Log(3, fmt.Sprintf("Statup Database could not be migrated: %v", tx.Error))
|
||||||
|
|
|
@ -51,7 +51,7 @@ func SelectService(id int64) *Service {
|
||||||
|
|
||||||
func (s *Service) Checkins() []*types.Checkin {
|
func (s *Service) Checkins() []*types.Checkin {
|
||||||
var hits []*types.Checkin
|
var hits []*types.Checkin
|
||||||
servicesDB().Where("service = ?", s.Id).Scan(&hits)
|
servicesDB().Where("service = ?", s.Id).Find(&hits)
|
||||||
return hits
|
return hits
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -89,8 +89,8 @@ var handlerFuncs = func(w http.ResponseWriter, r *http.Request) template.FuncMap
|
||||||
"js": func(html interface{}) template.JS {
|
"js": func(html interface{}) template.JS {
|
||||||
return template.JS(utils.ToString(html))
|
return template.JS(utils.ToString(html))
|
||||||
},
|
},
|
||||||
"safe": func(html interface{}) template.HTML {
|
"safe": func(html string) template.HTML {
|
||||||
return template.HTML(utils.ToString(html))
|
return template.HTML(html)
|
||||||
},
|
},
|
||||||
"Auth": func() bool {
|
"Auth": func() bool {
|
||||||
return IsAuthenticated(r)
|
return IsAuthenticated(r)
|
||||||
|
|
|
@ -47,7 +47,6 @@ func Router() *mux.Router {
|
||||||
r.PathPrefix("/statup.png").Handler(http.FileServer(source.TmplBox.HTTPBox()))
|
r.PathPrefix("/statup.png").Handler(http.FileServer(source.TmplBox.HTTPBox()))
|
||||||
}
|
}
|
||||||
r.PathPrefix("/js/").Handler(http.StripPrefix("/js/", http.FileServer(source.JsBox.HTTPBox())))
|
r.PathPrefix("/js/").Handler(http.StripPrefix("/js/", http.FileServer(source.JsBox.HTTPBox())))
|
||||||
//r.Handle("/charts/{id}.js", http.HandlerFunc(renderServiceChartHandler))
|
|
||||||
r.Handle("/charts.js", http.HandlerFunc(renderServiceChartsHandler))
|
r.Handle("/charts.js", http.HandlerFunc(renderServiceChartsHandler))
|
||||||
r.Handle("/setup", http.HandlerFunc(setupHandler)).Methods("GET")
|
r.Handle("/setup", http.HandlerFunc(setupHandler)).Methods("GET")
|
||||||
r.Handle("/setup", http.HandlerFunc(processSetupHandler)).Methods("POST")
|
r.Handle("/setup", http.HandlerFunc(processSetupHandler)).Methods("POST")
|
||||||
|
|
|
@ -0,0 +1,185 @@
|
||||||
|
// Statup
|
||||||
|
// Copyright (C) 2018. Hunter Long and the project contributors
|
||||||
|
// Written by Hunter Long <info@socialeck.com> and the project contributors
|
||||||
|
//
|
||||||
|
// https://github.com/hunterlong/statup
|
||||||
|
//
|
||||||
|
// 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 notifiers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"github.com/hunterlong/statup/core/notifier"
|
||||||
|
"github.com/hunterlong/statup/types"
|
||||||
|
"github.com/hunterlong/statup/utils"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
WEBHOOK_METHOD = "webhook"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Webhook struct {
|
||||||
|
*notifier.Notification
|
||||||
|
}
|
||||||
|
|
||||||
|
var webhook = &Webhook{¬ifier.Notification{
|
||||||
|
Method: WEBHOOK_METHOD,
|
||||||
|
Title: "HTTP Webhook",
|
||||||
|
Description: "Send a custom HTTP request to a specific URL with your own body, headers, and parameters",
|
||||||
|
Author: "Hunter Long",
|
||||||
|
AuthorUrl: "https://github.com/hunterlong",
|
||||||
|
Delay: time.Duration(1 * time.Second),
|
||||||
|
Form: []notifier.NotificationForm{{
|
||||||
|
Type: "text",
|
||||||
|
Title: "HTTP Endpoint",
|
||||||
|
Placeholder: "http://webhookurl.com/JW2MCP4SKQP",
|
||||||
|
SmallText: "Insert the URL for your HTTP Requests",
|
||||||
|
DbField: "Host",
|
||||||
|
Required: true,
|
||||||
|
}, {
|
||||||
|
Type: "text",
|
||||||
|
Title: "HTTP Method",
|
||||||
|
Placeholder: "POST",
|
||||||
|
SmallText: "Choose a HTTP method for example: GET, POST, DELETE, or PATCH",
|
||||||
|
DbField: "Var1",
|
||||||
|
Required: true,
|
||||||
|
}, {
|
||||||
|
Type: "textarea",
|
||||||
|
Title: "HTTP Body",
|
||||||
|
Placeholder: `{"service_id": "%s.Id", "service_name": "%s.Name"}`,
|
||||||
|
SmallText: "Optional HTTP body for a POST request. You can insert variables into your body request.<br>%service.Id, %service.Name<br>%failure.Issue",
|
||||||
|
DbField: "Var2",
|
||||||
|
}, {
|
||||||
|
Type: "text",
|
||||||
|
Title: "Content Type",
|
||||||
|
Placeholder: `application/json`,
|
||||||
|
SmallText: "Optional content type for example: application/json or text/plain",
|
||||||
|
DbField: "api_key",
|
||||||
|
}, {
|
||||||
|
Type: "text",
|
||||||
|
Title: "Header",
|
||||||
|
Placeholder: "Authorization=Token12345",
|
||||||
|
SmallText: "Optional Headers for request use format: KEY=Value,Key=Value",
|
||||||
|
DbField: "api_secret",
|
||||||
|
},
|
||||||
|
}}}
|
||||||
|
|
||||||
|
// DEFINE YOUR NOTIFICATION HERE.
|
||||||
|
func init() {
|
||||||
|
err := notifier.AddNotifier(webhook)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send will send a HTTP Post to the Webhook API. It accepts type: string
|
||||||
|
func (w *Webhook) Send(msg interface{}) error {
|
||||||
|
message := msg.(string)
|
||||||
|
_, err := w.run(message)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Webhook) Select() *notifier.Notification {
|
||||||
|
return w.Notification
|
||||||
|
}
|
||||||
|
|
||||||
|
func replaceBodyText(body string, s *types.Service, f *types.Failure) string {
|
||||||
|
if s != nil {
|
||||||
|
body = strings.Replace(body, "%service.Name", s.Name, -1)
|
||||||
|
body = strings.Replace(body, "%service.Id", utils.ToString(s.Id), -1)
|
||||||
|
}
|
||||||
|
if f != nil {
|
||||||
|
body = strings.Replace(body, "%failure.Issue", f.Issue, -1)
|
||||||
|
}
|
||||||
|
return body
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Webhook) run(body string) (*http.Response, error) {
|
||||||
|
utils.Log(1, fmt.Sprintf("sending body: '%v' to %v as a %v request", body, w.Host, w.Var1))
|
||||||
|
client := new(http.Client)
|
||||||
|
client.Timeout = time.Duration(10 * time.Second)
|
||||||
|
var buf *bytes.Buffer
|
||||||
|
buf = bytes.NewBuffer(nil)
|
||||||
|
if w.Var2 != "" {
|
||||||
|
buf = bytes.NewBuffer([]byte(w.Var2))
|
||||||
|
}
|
||||||
|
req, err := http.NewRequest(w.Var1, w.Host, buf)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if w.ApiSecret != "" {
|
||||||
|
splitArray := strings.Split(w.ApiSecret, ",")
|
||||||
|
for _, a := range splitArray {
|
||||||
|
split := strings.Split(a, "=")
|
||||||
|
req.Header.Add(split[0], split[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if w.ApiSecret != "" {
|
||||||
|
req.Header.Add("Content-Type", w.ApiSecret)
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *Webhook) OnTest() error {
|
||||||
|
service := &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: "test example",
|
||||||
|
LastResponse: "<html>this is an example response</html>",
|
||||||
|
CreatedAt: time.Now().Add(-24 * time.Hour),
|
||||||
|
}
|
||||||
|
body := replaceBodyText(w.Var2, service, nil)
|
||||||
|
resp, err := w.run(body)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
content, err := ioutil.ReadAll(resp.Body)
|
||||||
|
utils.Log(1, fmt.Sprintf("webhook notifier received: '%v'", string(content)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnFailure will trigger failing service
|
||||||
|
func (w *Webhook) OnFailure(s *types.Service, f *types.Failure) {
|
||||||
|
msg := replaceBodyText(w.Var2, s, f)
|
||||||
|
webhook.AddQueue(msg)
|
||||||
|
w.Online = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnSuccess will trigger successful service
|
||||||
|
func (w *Webhook) OnSuccess(s *types.Service) {
|
||||||
|
if !w.Online {
|
||||||
|
msg := replaceBodyText(w.Var2, s, nil)
|
||||||
|
webhook.AddQueue(msg)
|
||||||
|
}
|
||||||
|
w.Online = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// OnSave triggers when this notifier has been saved
|
||||||
|
func (w *Webhook) OnSave() error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -0,0 +1,104 @@
|
||||||
|
// Statup
|
||||||
|
// Copyright (C) 2018. Hunter Long and the project contributors
|
||||||
|
// Written by Hunter Long <info@socialeck.com> and the project contributors
|
||||||
|
//
|
||||||
|
// https://github.com/hunterlong/statup
|
||||||
|
//
|
||||||
|
// 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 notifiers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/hunterlong/statup/core/notifier"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
WEBHOOK_URL = "https://jsonplaceholder.typicode.com/posts"
|
||||||
|
webhookMessage = `{ "title": "%service.Id", "body": "%service.Name", "userId": 19999 }`
|
||||||
|
fullMsg string
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
webhook.Host = WEBHOOK_URL
|
||||||
|
webhook.Var1 = "POST"
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestWebhookNotifier(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
currentCount = CountNotifiers()
|
||||||
|
|
||||||
|
t.Run("Load Webhook", func(t *testing.T) {
|
||||||
|
webhook.Host = WEBHOOK_URL
|
||||||
|
webhook.Delay = time.Duration(100 * time.Millisecond)
|
||||||
|
err := notifier.AddNotifier(webhook)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
assert.Equal(t, "Hunter Long", webhook.Author)
|
||||||
|
assert.Equal(t, WEBHOOK_URL, webhook.Host)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Load Webhook Notifier", func(t *testing.T) {
|
||||||
|
notifier.Load()
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Notifier Tester", func(t *testing.T) {
|
||||||
|
assert.True(t, webhook.CanTest())
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Replace Body Text", func(t *testing.T) {
|
||||||
|
fullMsg = replaceBodyText(webhookMessage, TestService, TestFailure)
|
||||||
|
assert.Equal(t, 78, len(fullMsg))
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Within Limits", func(t *testing.T) {
|
||||||
|
ok, err := webhook.WithinLimits()
|
||||||
|
assert.Nil(t, err)
|
||||||
|
assert.True(t, ok)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook OnFailure", func(t *testing.T) {
|
||||||
|
webhook.OnFailure(TestService, TestFailure)
|
||||||
|
assert.Len(t, webhook.Queue, 1)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Check Offline", func(t *testing.T) {
|
||||||
|
assert.False(t, webhook.Online)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook OnSuccess", func(t *testing.T) {
|
||||||
|
webhook.OnSuccess(TestService)
|
||||||
|
assert.Len(t, webhook.Queue, 2)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Check Back Online", func(t *testing.T) {
|
||||||
|
assert.True(t, webhook.Online)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook OnSuccess Again", func(t *testing.T) {
|
||||||
|
webhook.OnSuccess(TestService)
|
||||||
|
assert.Len(t, webhook.Queue, 2)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Send", func(t *testing.T) {
|
||||||
|
err := webhook.Send(fullMsg)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
assert.Len(t, webhook.Queue, 2)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Webhook Queue", func(t *testing.T) {
|
||||||
|
go notifier.Queue(webhook)
|
||||||
|
time.Sleep(5 * time.Second)
|
||||||
|
assert.Equal(t, WEBHOOK_URL, webhook.Host)
|
||||||
|
assert.Equal(t, 1, len(webhook.Queue))
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
|
@ -2,13 +2,19 @@
|
||||||
{{$n := .Select}}
|
{{$n := .Select}}
|
||||||
<form method="POST" class="{{underscore $n.Method }}" action="/settings/notifier/{{ $n.Method }}">
|
<form method="POST" class="{{underscore $n.Method }}" action="/settings/notifier/{{ $n.Method }}">
|
||||||
{{if $n.Title}}<h4>{{$n.Title}}</h4>{{end}}
|
{{if $n.Title}}<h4>{{$n.Title}}</h4>{{end}}
|
||||||
{{if $n.Description}}<p class="small text-muted">{{safe $n.Description}}</p>{{end}}
|
{{if $n.Description}}<p class="small text-muted">{{$n.Description}}</p>{{end}}
|
||||||
|
|
||||||
{{range .Form}}
|
{{range $n.Form}}
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="text-capitalize" for="{{underscore .Title}}">{{.Title}}</label>
|
<label class="text-capitalize" for="{{underscore .Title}}">{{.Title}}</label>
|
||||||
<input type="{{.Type}}" name="{{underscore .DbField}}" class="form-control" value="{{ $n.GetValue .DbField }}" id="{{underscore .Title}}" placeholder="{{.Placeholder}}" {{if .Required}}required{{end}}>
|
{{if eq .Type "textarea"}}
|
||||||
{{if .SmallText}}<small class="form-text text-muted">{{safe .SmallText}}</small>{{end}}
|
<textarea rows="3" class="form-control" name="{{underscore .DbField}}" id="{{underscore .Title}}">{{ $n.GetValue .DbField }}</textarea>
|
||||||
|
{{else}}
|
||||||
|
<input type="{{.Type}}" name="{{underscore .DbField}}" class="form-control" value="{{ $n.GetValue .DbField }}" id="{{underscore .Title}}" placeholder="{{.Placeholder}}" {{if .Required}}required{{end}}>
|
||||||
|
{{end}}
|
||||||
|
{{if .SmallText}}
|
||||||
|
<small class="form-text text-muted">{{safe .SmallText}}</small>
|
||||||
|
{{end}}
|
||||||
</div>
|
</div>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
|
||||||
|
@ -26,10 +32,10 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="col-3 col-sm-2 mt-1">
|
<div class="col-3 col-sm-2 mt-1">
|
||||||
<span class="switch">
|
<span class="switch">
|
||||||
<input type="checkbox" name="enable" class="switch" id="switch-{{ $n.Method }}" {{if $n.Enabled}}checked{{end}}>
|
<input type="checkbox" name="enable" class="switch" id="switch-{{ $n.Method }}" {{if $n.Enabled}}checked{{end}}>
|
||||||
<label for="switch-{{ $n.Method }}"></label>
|
<label for="switch-{{ $n.Method }}"></label>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<input type="hidden" name="notifier" value="{{underscore $n.Method }}">
|
<input type="hidden" name="notifier" value="{{underscore $n.Method }}">
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
{{define "form_service"}}
|
{{define "form_service"}}
|
||||||
<form action="{{if ne .Id 0}}/service/{{.Id}}{{else}}/service{{end}}" method="POST">
|
<form action="{{if ne .Id 0}}/service/{{.Id}}{{else}}/services{{end}}" method="POST">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label for="service_name" class="col-sm-4 col-form-label">Service Name</label>
|
<label for="service_name" class="col-sm-4 col-form-label">Service Name</label>
|
||||||
<div class="col-sm-8">
|
<div class="col-sm-8">
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
{{define "form_user"}}
|
{{define "form_user"}}
|
||||||
<form action="{{if ne .Id 0}}/user/{{.Id}}{{else}}/user{{end}}" method="POST">
|
<form action="{{if ne .Id 0}}/user/{{.Id}}{{else}}/users{{end}}" method="POST">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label for="username" class="col-sm-4 col-form-label">Username</label>
|
<label for="username" class="col-sm-4 col-form-label">Username</label>
|
||||||
<div class="col-6 col-md-4">
|
<div class="col-6 col-md-4">
|
||||||
|
|
Loading…
Reference in New Issue