mirror of
https://github.com/gotenberg/gotenberg.git
synced 2026-07-02 00:17:40 +08:00
143 lines
3.8 KiB
Go
143 lines
3.8 KiB
Go
package webhook
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"log/slog"
|
|
"net/http"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/hashicorp/go-retryablehttp"
|
|
"github.com/labstack/echo/v4"
|
|
"go.opentelemetry.io/otel"
|
|
"go.opentelemetry.io/otel/codes"
|
|
"go.opentelemetry.io/otel/propagation"
|
|
"go.opentelemetry.io/otel/trace"
|
|
|
|
"github.com/gotenberg/gotenberg/v8/pkg/gotenberg"
|
|
)
|
|
|
|
// client gathers all the data required to send a request to a webhook.
|
|
type client struct {
|
|
url string
|
|
method string
|
|
errorUrl string
|
|
errorMethod string
|
|
extraHttpHeaders map[string]string
|
|
startTime time.Time
|
|
|
|
client *retryablehttp.Client
|
|
logger *slog.Logger
|
|
}
|
|
|
|
// send call the webhook either to send the success response or the error response.
|
|
func (c client) send(ctx context.Context, body io.Reader, headers map[string]string, errored bool) error {
|
|
url := c.url
|
|
if errored {
|
|
url = c.errorUrl
|
|
}
|
|
|
|
method := c.method
|
|
if errored {
|
|
method = c.errorMethod
|
|
}
|
|
|
|
spanName := fmt.Sprintf("%s Webhook", method)
|
|
if errored {
|
|
spanName = fmt.Sprintf("%s Webhook Error", method)
|
|
}
|
|
|
|
tracer := gotenberg.Tracer()
|
|
ctx, span := tracer.Start(ctx, spanName, trace.WithSpanKind(trace.SpanKindClient))
|
|
defer span.End()
|
|
|
|
req, err := retryablehttp.NewRequest(method, url, body)
|
|
if err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return fmt.Errorf("create '%s' request to '%s': %w", method, url, err)
|
|
}
|
|
|
|
// Inject trace context into outbound request headers.
|
|
otel.GetTextMapPropagator().Inject(ctx, propagation.HeaderCarrier(req.Header))
|
|
|
|
req.Header.Set("User-Agent", "Gotenberg")
|
|
|
|
// Extra HTTP headers are the custom headers from the user.
|
|
for key, value := range c.extraHttpHeaders {
|
|
req.Header.Set(key, value)
|
|
}
|
|
|
|
// Middleware caller's headers > extra HTTP headers from the user.
|
|
|
|
contentLength, ok := headers[echo.HeaderContentLength]
|
|
if ok {
|
|
// Golang "http" package should automatically calculate the size of the
|
|
// body. But when using a buffered file reader, it does not work.
|
|
// Worse, the "Content-Length" header is also removed. Therefore,
|
|
// to keep this valuable information, we have to trust the caller
|
|
// by reading the value of the "Content-Length" entry and set it as the
|
|
// content length of the request. It's kinda suboptimal, but hey, at
|
|
// least it works.
|
|
|
|
bodySize, err := strconv.ParseInt(contentLength, 10, 64)
|
|
if err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return fmt.Errorf("parse content length entry: %w", err)
|
|
}
|
|
|
|
req.ContentLength = bodySize
|
|
}
|
|
|
|
for key, value := range headers {
|
|
req.Header.Set(key, value)
|
|
}
|
|
|
|
resp, err := c.client.Do(req)
|
|
if err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return fmt.Errorf("send '%s' request to '%s': %w", method, url, err)
|
|
}
|
|
|
|
if resp.StatusCode >= http.StatusBadRequest {
|
|
err := fmt.Errorf("send '%s' request to '%s': got status: '%s'", method, url, resp.Status)
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return err
|
|
}
|
|
|
|
defer func() {
|
|
err := resp.Body.Close()
|
|
if err != nil {
|
|
c.logger.ErrorContext(ctx, fmt.Sprintf("close response body from '%s': %s", url, err))
|
|
}
|
|
}()
|
|
|
|
// Last piece for calculating the latency.
|
|
finishTime := time.Now()
|
|
|
|
// Now let's log!
|
|
attrs := []any{
|
|
slog.String("webhook_url", url),
|
|
slog.String("method", method),
|
|
slog.Int64("latency", int64(finishTime.Sub(c.startTime))),
|
|
slog.String("latency_human", finishTime.Sub(c.startTime).String()),
|
|
slog.Int64("bytes_out", req.ContentLength),
|
|
}
|
|
|
|
if errored {
|
|
c.logger.WarnContext(ctx, "request to webhook with error details handled", attrs...)
|
|
span.SetStatus(codes.Ok, "")
|
|
return nil
|
|
}
|
|
|
|
c.logger.InfoContext(ctx, "request to webhook handled", attrs...)
|
|
span.SetStatus(codes.Ok, "")
|
|
|
|
return nil
|
|
}
|