mirror of
https://git.mirrors.martin98.com/https://github.com/SigNoz/signoz
synced 2025-06-04 11:25:52 +08:00

### Summary Integrate the new implementations of the alertmanager along with changes to the ruler. This change can be broadly categoried into 3 parts: #### Frontend - The earlier `/api/v1/alerts` api was double encoding the response in json and sending it to the frontend. This PR fixes the json response object. For instance, we have gone from the response `{ "status": "success", "data": "{\"status\":\"success\",\"data\":[{\"labels\":{\"alertname\":\"[platform][consumer] consumer is above 100% memory utilization\",\"bu\":\"platform\",\"...... }` to the response `{"status":"success","data":[{"labels":{"alertname":"[Metrics] Pod CP......` - `msteams` has been changed to `msteamsv2` wherever applicable #### Ruler The following changes have been done in the ruler component: - Removal of the old alertmanager and notifier - The RuleDB methods `Create`, `Edit` and `Delete` have been made transactional - Introduction of a new `testPrepareNotifyFunc` for sending test notifications - Integration with the new alertmanager #### Alertmanager Although a huge chunk of the alertmanagers have been merged in previous PRs (the list can be found at https://github.com/SigNoz/platform-pod/issues/404), this PR takes care of changes needed in order to incorporate it with the ruler - Addition of ruleId based matching - Support for marshalling the global configuration directly from the upstream alertmanager - Addition of orgId to the legacy alertmanager - Support for always adding defaults to both routes and receivers while creating them - Migration to create the required alertmanager tables - Migration for msteams to msteamsv2 has been added. We will start using msteamv2 config for the new alertmanager and keep using msteams for the old one. #### Related Issues / PR's Closes https://github.com/SigNoz/platform-pod/issues/404 Closes https://github.com/SigNoz/platform-pod/issues/176
129 lines
3.4 KiB
Go
129 lines
3.4 KiB
Go
package middleware
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
)
|
|
|
|
const (
|
|
maxResponseBodyInLogs = 4096 // At most 4k bytes from response bodies in our logs.
|
|
)
|
|
|
|
type badResponseLoggingWriter interface {
|
|
http.ResponseWriter
|
|
// Get the status code.
|
|
StatusCode() int
|
|
// Get the error while writing.
|
|
WriteError() error
|
|
}
|
|
|
|
func newBadResponseLoggingWriter(rw http.ResponseWriter, buffer io.Writer) badResponseLoggingWriter {
|
|
b := nonFlushingBadResponseLoggingWriter{
|
|
rw: rw,
|
|
buffer: buffer,
|
|
logBody: false,
|
|
bodyBytesLeft: maxResponseBodyInLogs,
|
|
statusCode: http.StatusOK,
|
|
}
|
|
|
|
if f, ok := rw.(http.Flusher); ok {
|
|
return &flushingBadResponseLoggingWriter{b, f}
|
|
}
|
|
|
|
return &b
|
|
}
|
|
|
|
type nonFlushingBadResponseLoggingWriter struct {
|
|
rw http.ResponseWriter
|
|
buffer io.Writer
|
|
logBody bool
|
|
bodyBytesLeft int
|
|
statusCode int
|
|
writeError error // The error returned when downstream Write() fails.
|
|
}
|
|
|
|
// Extends nonFlushingBadResponseLoggingWriter that implements http.Flusher
|
|
type flushingBadResponseLoggingWriter struct {
|
|
nonFlushingBadResponseLoggingWriter
|
|
f http.Flusher
|
|
}
|
|
|
|
// Unwrap method is used by http.ResponseController to get access to original http.ResponseWriter.
|
|
func (writer *nonFlushingBadResponseLoggingWriter) Unwrap() http.ResponseWriter {
|
|
return writer.rw
|
|
}
|
|
|
|
// Header returns the header map that will be sent by WriteHeader.
|
|
// Implements ResponseWriter.
|
|
func (writer *nonFlushingBadResponseLoggingWriter) Header() http.Header {
|
|
return writer.rw.Header()
|
|
}
|
|
|
|
// WriteHeader writes the HTTP response header.
|
|
func (writer *nonFlushingBadResponseLoggingWriter) WriteHeader(statusCode int) {
|
|
writer.statusCode = statusCode
|
|
if statusCode >= 500 || statusCode == 400 {
|
|
writer.logBody = true
|
|
}
|
|
writer.rw.WriteHeader(statusCode)
|
|
}
|
|
|
|
// Writes HTTP response data.
|
|
func (writer *nonFlushingBadResponseLoggingWriter) Write(data []byte) (int, error) {
|
|
if writer.statusCode == 0 {
|
|
// WriteHeader has (probably) not been called, so we need to call it with StatusOK to fulfill the interface contract.
|
|
// https://godoc.org/net/http#ResponseWriter
|
|
writer.WriteHeader(http.StatusOK)
|
|
}
|
|
|
|
// 204 No Content is a success response that indicates that the request has been successfully processed and that the response body is intentionally empty.
|
|
if writer.statusCode == 204 {
|
|
return 0, nil
|
|
}
|
|
|
|
n, err := writer.rw.Write(data)
|
|
if writer.logBody {
|
|
writer.captureResponseBody(data)
|
|
}
|
|
if err != nil {
|
|
writer.writeError = err
|
|
}
|
|
return n, err
|
|
}
|
|
|
|
// Hijack hijacks the first response writer that is a Hijacker.
|
|
func (writer *nonFlushingBadResponseLoggingWriter) Hijack() (net.Conn, *bufio.ReadWriter, error) {
|
|
hj, ok := writer.rw.(http.Hijacker)
|
|
if ok {
|
|
return hj.Hijack()
|
|
}
|
|
return nil, nil, fmt.Errorf("cannot cast underlying response writer to Hijacker")
|
|
}
|
|
|
|
func (writer *nonFlushingBadResponseLoggingWriter) StatusCode() int {
|
|
return writer.statusCode
|
|
}
|
|
|
|
func (writer *nonFlushingBadResponseLoggingWriter) WriteError() error {
|
|
return writer.writeError
|
|
}
|
|
|
|
func (writer *flushingBadResponseLoggingWriter) Flush() {
|
|
writer.f.Flush()
|
|
}
|
|
|
|
func (writer *nonFlushingBadResponseLoggingWriter) captureResponseBody(data []byte) {
|
|
if len(data) > writer.bodyBytesLeft {
|
|
_, _ = writer.buffer.Write(data[:writer.bodyBytesLeft])
|
|
_, _ = io.WriteString(writer.buffer, "...")
|
|
writer.bodyBytesLeft = 0
|
|
writer.logBody = false
|
|
} else {
|
|
_, _ = writer.buffer.Write(data)
|
|
writer.bodyBytesLeft -= len(data)
|
|
}
|
|
}
|