2024-12-16 12:05:27 +00:00
|
|
|
// Copyright 2024 Woodpecker Authors
|
2024-11-30 12:57:59 +00:00
|
|
|
//
|
|
|
|
// 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"
|
2024-12-16 12:05:27 +00:00
|
|
|
"errors"
|
2024-11-30 12:57:59 +00:00
|
|
|
"fmt"
|
2024-12-16 12:05:27 +00:00
|
|
|
"net/http"
|
2024-11-30 12:57:59 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/urfave/cli/v3"
|
|
|
|
|
2024-12-22 09:44:34 +00:00
|
|
|
"go.woodpecker-ci.org/woodpecker/v3/cli/internal"
|
|
|
|
shared_utils "go.woodpecker-ci.org/woodpecker/v3/shared/utils"
|
|
|
|
"go.woodpecker-ci.org/woodpecker/v3/woodpecker-go/woodpecker"
|
2024-11-30 12:57:59 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
//nolint:mnd
|
|
|
|
var pipelinePurgeCmd = &cli.Command{
|
|
|
|
Name: "purge",
|
|
|
|
Usage: "purge pipelines",
|
|
|
|
ArgsUsage: "<repo-id|repo-full-name>",
|
|
|
|
Action: Purge,
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: "older-than",
|
|
|
|
Usage: "remove pipelines older than the specified time limit",
|
|
|
|
Required: true,
|
|
|
|
},
|
|
|
|
&cli.IntFlag{
|
|
|
|
Name: "keep-min",
|
|
|
|
Usage: "minimum number of pipelines to keep",
|
|
|
|
Value: 10,
|
|
|
|
},
|
|
|
|
&cli.BoolFlag{
|
|
|
|
Name: "dry-run",
|
|
|
|
Usage: "disable non-read api calls",
|
|
|
|
Value: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func Purge(ctx context.Context, c *cli.Command) error {
|
|
|
|
client, err := internal.NewClient(ctx, c)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return pipelinePurge(c, client)
|
|
|
|
}
|
|
|
|
|
|
|
|
func pipelinePurge(c *cli.Command, client woodpecker.Client) (err error) {
|
|
|
|
repoIDOrFullName := c.Args().First()
|
|
|
|
if len(repoIDOrFullName) == 0 {
|
|
|
|
return fmt.Errorf("missing required argument repo-id / repo-full-name")
|
|
|
|
}
|
|
|
|
repoID, err := internal.ParseRepo(client, repoIDOrFullName)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("invalid repo '%s': %w", repoIDOrFullName, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
olderThan := c.String("older-than")
|
|
|
|
keepMin := c.Int("keep-min")
|
|
|
|
dryRun := c.Bool("dry-run")
|
|
|
|
|
|
|
|
duration, err := time.ParseDuration(olderThan)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var pipelinesKeep []*woodpecker.Pipeline
|
|
|
|
|
|
|
|
if keepMin > 0 {
|
|
|
|
pipelinesKeep, err = fetchPipelinesToKeep(client, repoID, int(keepMin))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pipelines, err := fetchPipelines(client, repoID, duration)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a map of pipeline IDs to keep
|
|
|
|
keepMap := make(map[int64]struct{})
|
|
|
|
for _, p := range pipelinesKeep {
|
2024-12-15 06:20:08 +00:00
|
|
|
keepMap[p.Number] = struct{}{}
|
2024-11-30 12:57:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Filter pipelines to only include those not in keepMap
|
|
|
|
var pipelinesToPurge []*woodpecker.Pipeline
|
|
|
|
for _, p := range pipelines {
|
2024-12-15 06:20:08 +00:00
|
|
|
if _, exists := keepMap[p.Number]; !exists {
|
2024-11-30 12:57:59 +00:00
|
|
|
pipelinesToPurge = append(pipelinesToPurge, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
msgPrefix := ""
|
|
|
|
if dryRun {
|
|
|
|
msgPrefix = "DRY-RUN: "
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, p := range pipelinesToPurge {
|
2024-11-30 17:00:11 +00:00
|
|
|
// cspell:words spurge
|
2024-12-15 06:20:08 +00:00
|
|
|
log.Debug().Msgf("%spurge %v/%v pipelines from repo '%v' (pipeline %v)", msgPrefix, i+1, len(pipelinesToPurge), repoIDOrFullName, p.Number)
|
2024-11-30 12:57:59 +00:00
|
|
|
if dryRun {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-12-15 06:20:08 +00:00
|
|
|
err := client.PipelineDelete(repoID, p.Number)
|
2024-11-30 12:57:59 +00:00
|
|
|
if err != nil {
|
2024-12-16 12:05:27 +00:00
|
|
|
var clientErr *woodpecker.ClientError
|
|
|
|
if errors.As(err, &clientErr) && clientErr.StatusCode == http.StatusUnprocessableEntity {
|
|
|
|
log.Error().Err(err).Msgf("failed to delete pipeline %d", p.Number)
|
|
|
|
continue
|
|
|
|
}
|
2024-11-30 12:57:59 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func fetchPipelinesToKeep(client woodpecker.Client, repoID int64, keepMin int) ([]*woodpecker.Pipeline, error) {
|
|
|
|
if keepMin <= 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return shared_utils.Paginate(func(page int) ([]*woodpecker.Pipeline, error) {
|
|
|
|
return client.PipelineList(repoID,
|
|
|
|
woodpecker.PipelineListOptions{
|
|
|
|
ListOptions: woodpecker.ListOptions{
|
|
|
|
Page: page,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}, keepMin)
|
|
|
|
}
|
|
|
|
|
|
|
|
func fetchPipelines(client woodpecker.Client, repoID int64, duration time.Duration) ([]*woodpecker.Pipeline, error) {
|
|
|
|
return shared_utils.Paginate(func(page int) ([]*woodpecker.Pipeline, error) {
|
|
|
|
return client.PipelineList(repoID,
|
|
|
|
woodpecker.PipelineListOptions{
|
|
|
|
ListOptions: woodpecker.ListOptions{
|
|
|
|
Page: page,
|
|
|
|
},
|
2024-12-15 06:20:08 +00:00
|
|
|
Before: time.Now().Add(-duration),
|
2024-11-30 12:57:59 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}, -1)
|
|
|
|
}
|