2022-07-16 06:43:48 +02:00
package matrix
import (
"bytes"
2022-10-20 21:48:51 +02:00
"encoding/json"
2022-07-16 06:43:48 +02:00
"fmt"
"io"
"math/rand"
"net/http"
"net/url"
"time"
2022-12-06 07:41:09 +01:00
"github.com/TwiN/gatus/v5/alerting/alert"
"github.com/TwiN/gatus/v5/client"
"github.com/TwiN/gatus/v5/core"
2022-07-16 06:43:48 +02:00
)
// 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 {
2022-10-20 21:48:51 +02:00
buffer := bytes . NewBuffer ( provider . buildRequestBody ( endpoint , alert , result , resolved ) )
2022-07-16 06:43:48 +02:00
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
}
2022-10-20 21:16:27 +02:00
defer response . Body . Close ( )
2022-07-16 06:43:48 +02:00
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
}
2022-10-20 21:48:51 +02:00
type Body struct {
MsgType string ` json:"msgtype" `
Format string ` json:"format" `
Body string ` json:"body" `
FormattedBody string ` json:"formatted_body" `
}
2022-07-16 06:43:48 +02:00
// buildRequestBody builds the request body for the provider
2022-10-20 21:48:51 +02:00
func ( provider * AlertProvider ) buildRequestBody ( endpoint * core . Endpoint , alert * alert . Alert , result * core . Result , resolved bool ) [ ] byte {
body , _ := json . Marshal ( Body {
MsgType : "m.text" ,
Format : "org.matrix.custom.html" ,
Body : buildPlaintextMessageBody ( endpoint , alert , result , resolved ) ,
FormattedBody : buildHTMLMessageBody ( endpoint , alert , result , resolved ) ,
} )
return body
2022-07-16 06:43:48 +02:00
}
// 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 = "✕"
}
2022-10-20 21:48:51 +02:00
results += fmt . Sprintf ( "\n%s - %s" , prefix , conditionResult . Condition )
2022-07-16 06:43:48 +02:00
}
var description string
if alertDescription := alert . GetDescription ( ) ; len ( alertDescription ) > 0 {
2022-10-20 21:48:51 +02:00
description = "\n" + alertDescription
2022-07-16 06:43:48 +02:00
}
2022-10-20 21:48:51 +02:00
return fmt . Sprintf ( "%s%s\n%s" , message , description , results )
2022-07-16 06:43:48 +02:00
}
// 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 {
2022-10-20 21:48:51 +02:00
description = fmt . Sprintf ( "\n<blockquote>%s</blockquote>" , alertDescription )
2022-07-16 06:43:48 +02:00
}
2022-10-20 21:48:51 +02:00
return fmt . Sprintf ( "<h3>%s</h3>%s\n<h5>Condition results</h5><ul>%s</ul>" , message , description , results )
2022-07-16 06:43:48 +02:00
}
// 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-09-20 04:08:18 +02:00
return override . MatrixProviderConfig
2022-07-16 06:43:48 +02:00
}
}
}
2022-09-20 04:08:18 +02:00
return provider . MatrixProviderConfig
2022-07-16 06:43:48 +02:00
}
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
}
return string ( b )
}
// GetDefaultAlert returns the provider's default alert configuration
func ( provider AlertProvider ) GetDefaultAlert ( ) * alert . Alert {
return provider . DefaultAlert
}