mirror of
https://github.com/project-zot/zot.git
synced 2026-06-18 05:28:07 +08:00
da426850e7
* chore: Update golangci-lint Signed-off-by: Lars Francke <git@lars-francke.de> * chore: fix all golangci-lint issues - Remove deprecated `// +build` tags - Fix godoclint, modernize, wsl_v5, govet, lll, gci, noctx issues - Update linter configuration - Modernize code to use Go 1.22+ features (for range N, slices.Contains, etc.) - Update make check lint the privileged tests Signed-off-by: Andrei Aaron <andreifdaaron@gmail.com> --------- Signed-off-by: Lars Francke <git@lars-francke.de> Signed-off-by: Andrei Aaron <andreifdaaron@gmail.com> Co-authored-by: Lars Francke <git@lars-francke.de>
142 lines
3.2 KiB
Go
142 lines
3.2 KiB
Go
//go:build events
|
|
|
|
package events
|
|
|
|
import (
|
|
"context"
|
|
"encoding/base64"
|
|
"net/http"
|
|
"net/url"
|
|
|
|
cloudevents "github.com/cloudevents/sdk-go/v2"
|
|
cehttp "github.com/cloudevents/sdk-go/v2/protocol/http"
|
|
|
|
zerr "zotregistry.dev/zot/v2/errors"
|
|
eventsconf "zotregistry.dev/zot/v2/pkg/extensions/config/events"
|
|
)
|
|
|
|
type HTTPSink struct {
|
|
cloudevents.Client
|
|
|
|
config eventsconf.SinkConfig
|
|
}
|
|
|
|
func NewHTTPSink(config eventsconf.SinkConfig) (*HTTPSink, error) {
|
|
if config.Type != eventsconf.HTTP {
|
|
return nil, zerr.ErrInvalidEventSinkType
|
|
}
|
|
|
|
if config.Address == "" {
|
|
return nil, zerr.ErrEventSinkAddressEmpty
|
|
}
|
|
|
|
// Create the basic http client
|
|
httpClient, err := GetHTTPClientForConfig(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
opts := []cehttp.Option{
|
|
cehttp.WithTarget(config.Address),
|
|
cehttp.WithClient(*httpClient),
|
|
}
|
|
|
|
if config.Credentials != nil {
|
|
if config.Credentials.Username != "" {
|
|
opts = append(opts, cehttp.WithHeader("Authorization",
|
|
"Basic "+BasicAuth(config.Credentials.Username, config.Credentials.Password)))
|
|
} else if config.Credentials.Token != "" {
|
|
opts = append(opts, cehttp.WithHeader("Authorization",
|
|
"Bearer "+config.Credentials.Token))
|
|
}
|
|
}
|
|
|
|
if config.Headers != nil {
|
|
for key, value := range config.Headers {
|
|
opts = append(opts, cehttp.WithHeader(key, value))
|
|
}
|
|
}
|
|
|
|
// Create CloudEvents HTTP protocol
|
|
provider, err := cehttp.New(opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Create CloudEvents client
|
|
ceClient, err := cloudevents.NewClient(provider)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &HTTPSink{
|
|
Client: ceClient,
|
|
config: config,
|
|
}, nil
|
|
}
|
|
|
|
// Emit sends the event to the sink.
|
|
func (s *HTTPSink) Emit(event *cloudevents.Event) cloudevents.Result {
|
|
ctx, cancel := context.WithTimeout(context.Background(), s.config.Timeout)
|
|
defer cancel()
|
|
|
|
if err := event.Validate(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if s.config.Channel != "" {
|
|
event.SetExtension("channel", s.config.Channel)
|
|
}
|
|
|
|
// Send the event
|
|
return s.Send(ctx, *event)
|
|
}
|
|
|
|
// Close implements a method to clean up resources.
|
|
func (s *HTTPSink) Close() error {
|
|
// For HTTP clients, typically no specific cleanup is needed
|
|
// We could cancel any in-flight requests if we tracked them
|
|
return nil
|
|
}
|
|
|
|
func GetHTTPClientForConfig(config eventsconf.SinkConfig) (*http.Client, error) {
|
|
transport, ok := http.DefaultTransport.(*http.Transport)
|
|
if !ok {
|
|
return nil, zerr.ErrCouldNotCreateHTTPEventTransport
|
|
}
|
|
transport = transport.Clone()
|
|
|
|
if config.Proxy != nil && *config.Proxy != "" {
|
|
proxyURL, err := url.Parse(*config.Proxy)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
transport.Proxy = http.ProxyURL(proxyURL)
|
|
}
|
|
|
|
if config.TLSConfig != nil && (config.TLSConfig.CACertFile != "" || config.TLSConfig.CertFile != "") {
|
|
tlsConfig, err := getTLSConfig(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
transport.TLSClientConfig = tlsConfig
|
|
}
|
|
|
|
timeout := config.Timeout
|
|
if timeout == 0 {
|
|
timeout = DefaultHTTPTimeout
|
|
}
|
|
|
|
return &http.Client{
|
|
Transport: transport,
|
|
Timeout: timeout,
|
|
}, nil
|
|
}
|
|
|
|
// BasicAuth is a helper function for basic auth encoding.
|
|
func BasicAuth(username, password string) string {
|
|
auth := username + ":" + password
|
|
|
|
return base64.StdEncoding.EncodeToString([]byte(auth))
|
|
}
|