drone/internal/services/trigger/service.go

187 lines
5.3 KiB
Go

// Copyright 2023 Harness, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package trigger
import (
"context"
"errors"
"fmt"
"time"
"github.com/harness/gitness/events"
gitevents "github.com/harness/gitness/internal/events/git"
pullreqevents "github.com/harness/gitness/internal/events/pullreq"
"github.com/harness/gitness/internal/pipeline/commit"
"github.com/harness/gitness/internal/pipeline/triggerer"
"github.com/harness/gitness/internal/store"
"github.com/harness/gitness/stream"
"github.com/harness/gitness/types"
"github.com/harness/gitness/types/enum"
"github.com/hashicorp/go-multierror"
)
const (
eventsReaderGroupName = "gitness:trigger"
)
type Config struct {
EventReaderName string
Concurrency int
MaxRetries int
}
func (c *Config) Prepare() 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.MaxRetries < 0 {
return errors.New("config.MaxRetries can't be negative")
}
return nil
}
type Service struct {
triggerStore store.TriggerStore
pullReqStore store.PullReqStore
repoStore store.RepoStore
pipelineStore store.PipelineStore
triggerSvc triggerer.Triggerer
commitSvc commit.Service
}
func New(
ctx context.Context,
config Config,
triggerStore store.TriggerStore,
pullReqStore store.PullReqStore,
repoStore store.RepoStore,
pipelineStore store.PipelineStore,
triggerSvc triggerer.Triggerer,
commitSvc commit.Service,
gitReaderFactory *events.ReaderFactory[*gitevents.Reader],
pullreqEvReaderFactory *events.ReaderFactory[*pullreqevents.Reader],
) (*Service, error) {
if err := config.Prepare(); err != nil {
return nil, fmt.Errorf("provided trigger service config is invalid: %w", err)
}
service := &Service{
triggerStore: triggerStore,
pullReqStore: pullReqStore,
repoStore: repoStore,
commitSvc: commitSvc,
pipelineStore: pipelineStore,
triggerSvc: triggerSvc,
}
_, err := gitReaderFactory.Launch(ctx, eventsReaderGroupName, config.EventReaderName,
func(r *gitevents.Reader) error {
const idleTimeout = 1 * time.Minute
r.Configure(
stream.WithConcurrency(config.Concurrency),
stream.WithHandlerOptions(
stream.WithIdleTimeout(idleTimeout),
stream.WithMaxRetries(config.MaxRetries),
))
_ = r.RegisterBranchCreated(service.handleEventBranchCreated)
_ = r.RegisterBranchUpdated(service.handleEventBranchUpdated)
_ = r.RegisterTagCreated(service.handleEventTagCreated)
_ = r.RegisterTagUpdated(service.handleEventTagUpdated)
return nil
})
if err != nil {
return nil, fmt.Errorf("failed to launch git events reader: %w", err)
}
_, err = pullreqEvReaderFactory.Launch(ctx, eventsReaderGroupName, config.EventReaderName,
func(r *pullreqevents.Reader) error {
const idleTimeout = 1 * time.Minute
r.Configure(
stream.WithConcurrency(config.Concurrency),
stream.WithHandlerOptions(
stream.WithIdleTimeout(idleTimeout),
// retries not needed for builds which failed to trigger, can be adjusted when needed
stream.WithMaxRetries(0),
))
_ = r.RegisterCreated(service.handleEventPullReqCreated)
_ = r.RegisterBranchUpdated(service.handleEventPullReqBranchUpdated)
_ = r.RegisterReopened(service.handleEventPullReqReopened)
return nil
})
if err != nil {
return nil, fmt.Errorf("failed to launch pr events reader: %w", err)
}
return service, nil
}
// trigger a build given an action on a repo and a hook.
// It tries to find all enabled triggers, see if the action is the same
// as the trigger action - and if so, find the pipeline for the trigger
// and fire an execution.
func (s *Service) trigger(ctx context.Context, repoID int64,
action enum.TriggerAction, hook *triggerer.Hook) error {
// Get all enabled triggers for a repo.
ret, err := s.triggerStore.ListAllEnabled(ctx, repoID)
if err != nil {
return fmt.Errorf("failed to list all enabled triggers: %w", err)
}
validTriggers := []*types.Trigger{}
// Check which triggers are eligible to be fired
for _, t := range ret {
for _, a := range t.Actions {
if a == action {
validTriggers = append(validTriggers, t)
break
}
}
}
var errs error
for _, t := range validTriggers {
// TODO: We can make a minor optimization here to not fetch a pipeline each time
// since there could be multiple triggers for a pipeline.
pipeline, err := s.pipelineStore.Find(ctx, t.PipelineID)
if err != nil {
errs = multierror.Append(errs, err)
continue
}
// Don't fire triggers for disabled pipelines
if pipeline.Disabled {
continue
}
_, err = s.triggerSvc.Trigger(ctx, pipeline, hook)
if err != nil {
errs = multierror.Append(errs, err)
}
}
return errs
}