drone/internal/services/webhook/service.go

116 lines
3.7 KiB
Go

// Copyright 2022 Harness Inc. All rights reserved.
// Use of this source code is governed by the Polyform Free Trial License
// that can be found in the LICENSE.md file for this repository.
package webhook
import (
"context"
"errors"
"fmt"
"net/http"
"time"
"github.com/harness/gitness/events"
"github.com/harness/gitness/gitrpc"
gitevents "github.com/harness/gitness/internal/events/git"
"github.com/harness/gitness/internal/store"
"github.com/harness/gitness/internal/url"
)
const (
eventsReaderGroupName = "webhook"
processingTimeout = 2 * time.Minute
)
type Config struct {
EventReaderName string `envconfig:"GITNESS_WEBHOOK_EVENT_READER_NAME"`
Concurrency int `envconfig:"GITNESS_WEBHOOK_CONCURRENCY" default:"4"`
MaxRetryCount int64 `envconfig:"GITNESS_WEBHOOK_MAX_RETRY_COUNT" default:"3"`
AllowPrivateNetwork bool `envconfig:"GITNESS_WEBHOOK_ALLOW_PRIVATE_NETWORK" default:"false"`
AllowLoopback bool `envconfig:"GITNESS_WEBHOOK_ALLOW_LOOPBACK" default:"false"`
}
func (c *Config) Validate() error {
if c == nil {
return errors.New("config is required")
}
if c.EventReaderName == "" {
return errors.New("config.EventReaderName is required")
}
if c.Concurrency < 1 {
return errors.New("config.Concurrency has to be a positive number")
}
if c.MaxRetryCount < 0 {
return errors.New("config.MaxRetryCount can't be negative")
}
return nil
}
// Service is responsible for processing webhook events.
type Service struct {
webhookStore store.WebhookStore
webhookExecutionStore store.WebhookExecutionStore
urlProvider *url.Provider
repoStore store.RepoStore
principalStore store.PrincipalStore
gitRPCClient gitrpc.Interface
readerCanceler *events.ReaderCanceler
secureHTTPClient *http.Client
insecureHTTPClient *http.Client
}
func NewService(ctx context.Context, config Config,
gitReaderFactory *events.ReaderFactory[*gitevents.Reader],
webhookStore store.WebhookStore, webhookExecutionStore store.WebhookExecutionStore,
repoStore store.RepoStore, urlProvider *url.Provider,
principalStore store.PrincipalStore, gitRPCClient gitrpc.Interface) (*Service, error) {
if err := config.Validate(); err != nil {
return nil, fmt.Errorf("provided config is invalid: %w", err)
}
service := &Service{
webhookStore: webhookStore,
webhookExecutionStore: webhookExecutionStore,
repoStore: repoStore,
urlProvider: urlProvider,
principalStore: principalStore,
gitRPCClient: gitRPCClient,
// set after launching factory
readerCanceler: nil,
secureHTTPClient: newHTTPClient(config.AllowLoopback, config.AllowPrivateNetwork, false),
insecureHTTPClient: newHTTPClient(config.AllowLoopback, config.AllowPrivateNetwork, true),
}
canceler, err := gitReaderFactory.Launch(ctx, eventsReaderGroupName, config.EventReaderName,
func(r *gitevents.Reader) error {
// configure reader
_ = r.SetConcurrency(config.Concurrency)
_ = r.SetMaxRetryCount(config.MaxRetryCount)
_ = r.SetProcessingTimeout(processingTimeout)
// register events
_ = r.RegisterBranchCreated(service.handleEventBranchCreated)
_ = r.RegisterBranchUpdated(service.handleEventBranchUpdated)
_ = r.RegisterBranchDeleted(service.handleEventBranchDeleted)
_ = r.RegisterTagCreated(service.handleEventTagCreated)
_ = r.RegisterTagUpdated(service.handleEventTagUpdated)
_ = r.RegisterTagDeleted(service.handleEventTagDeleted)
return nil
})
if err != nil {
return nil, fmt.Errorf("failed to launch event reader for webhooks: %w", err)
}
service.readerCanceler = canceler
return service, nil
}
func (s *Service) Cancel() error {
return s.readerCanceler.Cancel()
}