2022-07-16 06:43:48 +02:00
package matrix
import (
"bytes"
"fmt"
"io"
"math/rand"
"net/http"
"net/url"
"time"
"github.com/TwiN/gatus/v4/alerting/alert"
"github.com/TwiN/gatus/v4/client"
"github.com/TwiN/gatus/v4/core"
)
// AlertProvider is the configuration necessary for sending an alert using Matrix
type AlertProvider struct {
2022-07-19 19:15:41 +02:00
MatrixProviderConfig ` yaml:",inline" `
2022-07-16 06:43:48 +02:00
// DefaultAlert is the default alert configuration to use for endpoints with an alert of the appropriate type
DefaultAlert * alert . Alert ` yaml:"default-alert,omitempty" `
// Overrides is a list of Override that may be prioritized over the default configuration
Overrides [ ] Override ` yaml:"overrides,omitempty" `
}
// Override is a case under which the default integration is overridden
type Override struct {
Group string ` yaml:"group" `
2022-07-19 19:15:41 +02:00
MatrixProviderConfig ` yaml:",inline" `
2022-07-16 06:43:48 +02:00
}
const defaultHomeserverURL = "https://matrix-client.matrix.org"
2022-07-19 19:15:41 +02:00
type MatrixProviderConfig struct {
// ServerURL is the custom homeserver to use (optional)
ServerURL string ` yaml:"server-url" `
// AccessToken is the bot user's access token to send messages
AccessToken string ` yaml:"access-token" `
// InternalRoomID is the room that the bot user has permissions to send messages to
2022-07-16 06:43:48 +02:00
InternalRoomID string ` yaml:"internal-room-id" `
}
// IsValid returns whether the provider's configuration is valid
func ( provider * AlertProvider ) IsValid ( ) bool {
registeredGroups := make ( map [ string ] bool )
if provider . Overrides != nil {
for _ , override := range provider . Overrides {
if isAlreadyRegistered := registeredGroups [ override . Group ] ; isAlreadyRegistered || override . Group == "" || len ( override . AccessToken ) == 0 || len ( override . InternalRoomID ) == 0 {
return false
}
registeredGroups [ override . Group ] = true
}
}
return len ( provider . AccessToken ) > 0 && len ( provider . InternalRoomID ) > 0
}
// Send an alert using the provider
func ( provider * AlertProvider ) Send ( endpoint * core . Endpoint , alert * alert . Alert , result * core . Result , resolved bool ) error {
buffer := bytes . NewBuffer ( [ ] byte ( provider . buildRequestBody ( endpoint , alert , result , resolved ) ) )
config := provider . getConfigForGroup ( endpoint . Group )
2022-07-19 19:15:41 +02:00
if config . ServerURL == "" {
config . ServerURL = defaultHomeserverURL
2022-07-16 06:43:48 +02:00
}
2022-07-19 19:42:28 +02:00
// The Matrix endpoint requires a unique transaction ID for each event sent
2022-07-16 06:43:48 +02:00
txnId := randStringBytes ( 24 )
request , err := http . NewRequest (
http . MethodPut ,
2022-07-19 19:40:07 +02:00
fmt . Sprintf ( "%s/_matrix/client/v3/rooms/%s/send/m.room.message/%s?access_token=%s" ,
2022-07-19 19:15:41 +02:00
config . ServerURL ,
2022-07-16 06:43:48 +02:00
url . PathEscape ( config . InternalRoomID ) ,
txnId ,
url . QueryEscape ( config . AccessToken ) ,
) ,
buffer ,
)
if err != nil {
return err
}
request . Header . Set ( "Content-Type" , "application/json" )
response , err := client . GetHTTPClient ( nil ) . Do ( request )
if err != nil {
return err
}
if response . StatusCode > 399 {
body , _ := io . ReadAll ( response . Body )
return fmt . Errorf ( "call to provider alert returned status code %d: %s" , response . StatusCode , string ( body ) )
}
return err
}
// buildRequestBody builds the request body for the provider
func ( provider * AlertProvider ) buildRequestBody ( endpoint * core . Endpoint , alert * alert . Alert , result * core . Result , resolved bool ) string {
return fmt . Sprintf ( ` {
"msgtype" : "m.text" ,
"format" : "org.matrix.custom.html" ,
"body" : "%s" ,
"formatted_body" : "%s"
} ` ,
buildPlaintextMessageBody ( endpoint , alert , result , resolved ) ,
buildHTMLMessageBody ( endpoint , alert , result , resolved ) ,
)
}
// buildPlaintextMessageBody builds the message body in plaintext to include in request
func buildPlaintextMessageBody ( endpoint * core . Endpoint , alert * alert . Alert , result * core . Result , resolved bool ) string {
var message , results string
if resolved {
message = fmt . Sprintf ( "An alert for `%s` has been resolved after passing successfully %d time(s) in a row" , endpoint . DisplayName ( ) , alert . SuccessThreshold )
} else {
message = fmt . Sprintf ( "An alert for `%s` has been triggered due to having failed %d time(s) in a row" , endpoint . DisplayName ( ) , alert . FailureThreshold )
}
for _ , conditionResult := range result . ConditionResults {
var prefix string
if conditionResult . Success {
prefix = "✓"
} else {
prefix = "✕"
}
results += fmt . Sprintf ( "\\n%s - %s" , prefix , conditionResult . Condition )
}
var description string
if alertDescription := alert . GetDescription ( ) ; len ( alertDescription ) > 0 {
description = "\\n" + alertDescription
}
return fmt . Sprintf ( "%s%s\\n%s" , message , description , results )
}
// buildHTMLMessageBody builds the message body in HTML to include in request
func buildHTMLMessageBody ( endpoint * core . Endpoint , alert * alert . Alert , result * core . Result , resolved bool ) string {
var message , results string
if resolved {
message = fmt . Sprintf ( "An alert for <code>%s</code> has been resolved after passing successfully %d time(s) in a row" , endpoint . DisplayName ( ) , alert . SuccessThreshold )
} else {
message = fmt . Sprintf ( "An alert for <code>%s</code> has been triggered due to having failed %d time(s) in a row" , endpoint . DisplayName ( ) , alert . FailureThreshold )
}
for _ , conditionResult := range result . ConditionResults {
var prefix string
if conditionResult . Success {
prefix = "✅"
} else {
prefix = "❌"
}
results += fmt . Sprintf ( "<li>%s - <code>%s</code></li>" , prefix , conditionResult . Condition )
}
var description string
if alertDescription := alert . GetDescription ( ) ; len ( alertDescription ) > 0 {
description = fmt . Sprintf ( "\\n<blockquote>%s</blockquote>" , alertDescription )
}
return fmt . Sprintf ( "<h3>%s</h3>%s\\n<h5>Condition results</h5><ul>%s</ul>" , message , description , results )
}
// getConfigForGroup returns the appropriate configuration for a given group
2022-07-19 19:15:41 +02:00
func ( provider * AlertProvider ) getConfigForGroup ( group string ) MatrixProviderConfig {
2022-07-16 06:43:48 +02:00
if provider . Overrides != nil {
for _ , override := range provider . Overrides {
if group == override . Group {
2022-07-19 19:15:41 +02:00
return MatrixProviderConfig {
ServerURL : override . ServerURL ,
2022-07-16 06:43:48 +02:00
AccessToken : override . AccessToken ,
InternalRoomID : override . InternalRoomID ,
}
}
}
}
2022-07-19 19:15:41 +02:00
return MatrixProviderConfig {
ServerURL : provider . ServerURL ,
2022-07-16 06:43:48 +02:00
AccessToken : provider . AccessToken ,
InternalRoomID : provider . InternalRoomID ,
}
}
func randStringBytes ( n int ) string {
2022-07-19 19:42:28 +02:00
// All the compatible characters to use in a transaction ID
const availableCharacterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"
2022-07-16 06:43:48 +02:00
b := make ( [ ] byte , n )
rand . Seed ( time . Now ( ) . UnixNano ( ) )
for i := range b {
2022-07-19 19:42:28 +02:00
b [ i ] = availableCharacterBytes [ rand . Intn ( len ( availableCharacterBytes ) ) ]
2022-07-16 06:43:48 +02:00
}
2022-07-19 19:42:28 +02:00
2022-07-16 06:43:48 +02:00
return string ( b )
}
// GetDefaultAlert returns the provider's default alert configuration
func ( provider AlertProvider ) GetDefaultAlert ( ) * alert . Alert {
return provider . DefaultAlert
}