mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2024-12-25 01:40:30 +00:00
16dca0abc2
This stores workflows and steps to DB even if it is not yet approved and thus blocked. I'm not really happy with this, because even though it is stored, it must parse the pipeline again and set back the original UUID. If you have any ideas how to fix/improve this just comment. In addition, this allows to view step list and side panel for approved pipelines, https://github.com/woodpecker-ci/woodpecker/pull/2345 is partially not longer necessary. Closes https://github.com/woodpecker-ci/woodpecker/issues/895 --------- Co-authored-by: 6543 <6543@obermui.de>
131 lines
4.3 KiB
Go
131 lines
4.3 KiB
Go
// Copyright 2022 Woodpecker Authors
|
|
//
|
|
// 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 pipeline
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"go.woodpecker-ci.org/woodpecker/v2/server"
|
|
forge_types "go.woodpecker-ci.org/woodpecker/v2/server/forge/types"
|
|
"go.woodpecker-ci.org/woodpecker/v2/server/model"
|
|
"go.woodpecker-ci.org/woodpecker/v2/server/store"
|
|
)
|
|
|
|
// Restart a pipeline by creating a new one out of the old and start it
|
|
func Restart(ctx context.Context, store store.Store, lastPipeline *model.Pipeline, user *model.User, repo *model.Repo, envs map[string]string) (*model.Pipeline, error) {
|
|
forge := server.Config.Services.Forge
|
|
switch lastPipeline.Status {
|
|
case model.StatusDeclined,
|
|
model.StatusBlocked:
|
|
return nil, &ErrBadRequest{Msg: fmt.Sprintf("cannot restart a pipeline with status %s", lastPipeline.Status)}
|
|
}
|
|
|
|
// fetch the old pipeline config from the database
|
|
configs, err := store.ConfigsForPipeline(lastPipeline.ID)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("failure to get pipeline config for %s", repo.FullName)
|
|
return nil, &ErrNotFound{Msg: fmt.Sprintf("failure to get pipeline config for %s. %s", repo.FullName, err)}
|
|
}
|
|
|
|
var pipelineFiles []*forge_types.FileMeta
|
|
for _, y := range configs {
|
|
pipelineFiles = append(pipelineFiles, &forge_types.FileMeta{Data: y.Data, Name: y.Name})
|
|
}
|
|
|
|
// If the config service is active we should refetch the config in case something changed
|
|
configService := server.Config.Services.Manager.ConfigServiceFromRepo(repo)
|
|
pipelineFiles, err = configService.Fetch(ctx, forge, user, repo, lastPipeline, pipelineFiles, true)
|
|
if err != nil {
|
|
return nil, &ErrBadRequest{
|
|
Msg: fmt.Sprintf("On fetching external pipeline config: %s", err),
|
|
}
|
|
}
|
|
|
|
newPipeline := createNewOutOfOld(lastPipeline)
|
|
newPipeline.Parent = lastPipeline.ID
|
|
|
|
err = store.CreatePipeline(newPipeline)
|
|
if err != nil {
|
|
msg := fmt.Sprintf("failure to save pipeline for %s", repo.FullName)
|
|
log.Error().Err(err).Msg(msg)
|
|
return nil, fmt.Errorf(msg)
|
|
}
|
|
|
|
if len(configs) == 0 {
|
|
newPipeline, uerr := UpdateToStatusError(store, *newPipeline, errors.New("pipeline definition not found"))
|
|
if uerr != nil {
|
|
log.Debug().Err(uerr).Msg("failure to update pipeline status")
|
|
} else {
|
|
updatePipelineStatus(ctx, newPipeline, repo, user)
|
|
}
|
|
return newPipeline, nil
|
|
}
|
|
if err := linkPipelineConfigs(store, configs, newPipeline.ID); err != nil {
|
|
msg := fmt.Sprintf("failure to persist pipeline config for %s.", repo.FullName)
|
|
log.Error().Err(err).Msg(msg)
|
|
return nil, fmt.Errorf(msg)
|
|
}
|
|
|
|
newPipeline, pipelineItems, err := createPipelineItems(ctx, store, newPipeline, user, repo, pipelineFiles, envs)
|
|
if err != nil {
|
|
msg := fmt.Sprintf("failure to createPipelineItems for %s", repo.FullName)
|
|
log.Error().Err(err).Msg(msg)
|
|
return nil, fmt.Errorf(msg)
|
|
}
|
|
|
|
if err := prepareStart(ctx, store, newPipeline, user, repo); err != nil {
|
|
msg := fmt.Sprintf("failure to prepare pipeline for %s", repo.FullName)
|
|
log.Error().Err(err).Msg(msg)
|
|
return nil, fmt.Errorf(msg)
|
|
}
|
|
|
|
newPipeline, err = start(ctx, store, newPipeline, user, repo, pipelineItems)
|
|
if err != nil {
|
|
msg := fmt.Sprintf("failure to start pipeline for %s", repo.FullName)
|
|
log.Error().Err(err).Msg(msg)
|
|
return nil, fmt.Errorf(msg)
|
|
}
|
|
|
|
return newPipeline, nil
|
|
}
|
|
|
|
func linkPipelineConfigs(store store.Store, configs []*model.Config, pipelineID int64) error {
|
|
for _, conf := range configs {
|
|
pipelineConfig := &model.PipelineConfig{
|
|
ConfigID: conf.ID,
|
|
PipelineID: pipelineID,
|
|
}
|
|
err := store.PipelineConfigCreate(pipelineConfig)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func createNewOutOfOld(old *model.Pipeline) *model.Pipeline {
|
|
newPipeline := *old
|
|
newPipeline.ID = 0
|
|
newPipeline.Number = 0
|
|
newPipeline.Status = model.StatusPending
|
|
newPipeline.Started = 0
|
|
newPipeline.Finished = 0
|
|
newPipeline.Errors = nil
|
|
return &newPipeline
|
|
}
|