2022-01-08 18:46:27 +01:00
|
|
|
package webhook
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
2024-04-16 22:51:47 -03:00
|
|
|
"github.com/caarlos0/env/v11"
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2024-05-26 15:02:57 -03:00
|
|
|
"github.com/goreleaser/goreleaser/v2/internal/tmpl"
|
|
|
|
"github.com/goreleaser/goreleaser/v2/pkg/context"
|
2022-01-08 18:46:27 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
defaultMessageTemplate = `{ "message": "{{ .ProjectName }} {{ .Tag }} is out! Check it out at {{ .ReleaseURL }}"}`
|
|
|
|
ContentTypeHeaderKey = "Content-Type"
|
|
|
|
UserAgentHeaderKey = "User-Agent"
|
2024-04-08 01:46:14 +03:00
|
|
|
UserAgentHeaderValue = "goreleaser"
|
2022-01-08 18:46:27 +01:00
|
|
|
AuthorizationHeaderKey = "Authorization"
|
|
|
|
DefaultContentType = "application/json; charset=utf-8"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Pipe struct{}
|
|
|
|
|
|
|
|
func (Pipe) String() string { return "webhook" }
|
|
|
|
func (Pipe) Skip(ctx *context.Context) bool { return !ctx.Config.Announce.Webhook.Enabled }
|
|
|
|
|
|
|
|
type Config struct {
|
|
|
|
BasicAuthHeader string `env:"BASIC_AUTH_HEADER_VALUE"`
|
|
|
|
BearerTokenHeader string `env:"BEARER_TOKEN_HEADER_VALUE"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p Pipe) Default(ctx *context.Context) error {
|
|
|
|
if ctx.Config.Announce.Webhook.MessageTemplate == "" {
|
|
|
|
ctx.Config.Announce.Webhook.MessageTemplate = defaultMessageTemplate
|
|
|
|
}
|
|
|
|
if ctx.Config.Announce.Webhook.ContentType == "" {
|
|
|
|
ctx.Config.Announce.Webhook.ContentType = DefaultContentType
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p Pipe) Announce(ctx *context.Context) error {
|
2024-04-16 22:51:47 -03:00
|
|
|
cfg, err := env.ParseAs[Config]()
|
|
|
|
if err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
endpointURLConfig, err := tmpl.New(ctx).Apply(ctx.Config.Announce.Webhook.EndpointURL)
|
|
|
|
if err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
if len(endpointURLConfig) == 0 {
|
2022-12-28 12:24:21 -03:00
|
|
|
return errors.New("webhook: no endpoint url")
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := url.ParseRequestURI(endpointURLConfig); err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
endpointURL, err := url.Parse(endpointURLConfig)
|
|
|
|
if err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
msg, err := tmpl.New(ctx).Apply(ctx.Config.Announce.Webhook.MessageTemplate)
|
|
|
|
if err != nil {
|
2023-08-24 22:06:12 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof("posting: '%s'", msg)
|
|
|
|
customTransport := http.DefaultTransport.(*http.Transport).Clone()
|
|
|
|
|
|
|
|
customTransport.TLSClientConfig = &tls.Config{
|
|
|
|
InsecureSkipVerify: ctx.Config.Announce.Webhook.SkipTLSVerify,
|
|
|
|
}
|
|
|
|
|
|
|
|
client := &http.Client{
|
|
|
|
Transport: customTransport,
|
|
|
|
}
|
|
|
|
|
|
|
|
req, err := http.NewRequest(http.MethodPost, endpointURL.String(), strings.NewReader(msg))
|
|
|
|
if err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
req.Header.Add(ContentTypeHeaderKey, ctx.Config.Announce.Webhook.ContentType)
|
|
|
|
req.Header.Add(UserAgentHeaderKey, UserAgentHeaderValue)
|
|
|
|
|
|
|
|
if cfg.BasicAuthHeader != "" {
|
|
|
|
log.Debugf("set basic auth header")
|
|
|
|
req.Header.Add(AuthorizationHeaderKey, cfg.BasicAuthHeader)
|
|
|
|
} else if cfg.BearerTokenHeader != "" {
|
|
|
|
log.Debugf("set bearer token header")
|
|
|
|
req.Header.Add(AuthorizationHeaderKey, cfg.BearerTokenHeader)
|
|
|
|
}
|
|
|
|
|
|
|
|
for key, value := range ctx.Config.Announce.Webhook.Headers {
|
|
|
|
log.Debugf("Header Key %s / Value %s", key, value)
|
|
|
|
req.Header.Add(key, value)
|
|
|
|
}
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
2022-12-28 12:24:21 -03:00
|
|
|
return fmt.Errorf("webhook: %w", err)
|
2022-01-08 18:46:27 +01:00
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
switch resp.StatusCode {
|
|
|
|
case http.StatusOK, http.StatusCreated, http.StatusAccepted, http.StatusNoContent:
|
|
|
|
log.Infof("Post OK: '%v'", resp.StatusCode)
|
|
|
|
body, _ := io.ReadAll(resp.Body)
|
|
|
|
log.Infof("Response : %v\n", string(body))
|
|
|
|
return nil
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("request failed with status %v", resp.Status)
|
|
|
|
}
|
|
|
|
}
|