Skip to content

Commit

Permalink
refactor: use httptest instead of mocking client
Browse files Browse the repository at this point in the history
Signed-off-by: KevFan <chfan@redhat.com>
  • Loading branch information
KevFan committed Sep 11, 2024
1 parent 5f359fe commit e5ea790
Show file tree
Hide file tree
Showing 3 changed files with 102 additions and 107 deletions.
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ require (
k8s.io/klog/v2 v2.120.1
k8s.io/utils v0.0.0-20240423183400-0849a56e8f22
maistra.io/istio-operator v0.0.0-20240217080932-98753cb28cd7
oras.land/oras-go v1.2.4
sigs.k8s.io/controller-runtime v0.18.0
sigs.k8s.io/external-dns v0.14.0
sigs.k8s.io/gateway-api v1.1.0
Expand Down Expand Up @@ -173,6 +172,7 @@ require (
k8s.io/component-base v0.30.0 // indirect
k8s.io/kube-openapi v0.0.0-20240423202451-8948a665c108 // indirect
k8s.io/kubectl v0.29.1 // indirect
oras.land/oras-go v1.2.4 // indirect
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect
sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3 // indirect
sigs.k8s.io/kustomize/kyaml v0.16.0 // indirect
Expand Down
58 changes: 32 additions & 26 deletions quay/quay_overflow.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,22 +13,23 @@ import (
"time"

"golang.org/x/exp/maps"
"oras.land/oras-go/pkg/registry/remote"
)

const (
// Max number of entries returned as specified in Quay API docs for listing tags
pageLimit = 100
pageLimit = 100
accessTokenEnvKey = "ACCESS_TOKEN"
)

var (
accessToken = os.Getenv("ACCESS_TOKEN")
accessToken = os.Getenv(accessTokenEnvKey)
preserveSubstrings = []string{
"latest",
// Preserve semver release branch or semver tag regex - release-vX.Y.Z(-rc1) or vX.Y.Z(-rc1)
// Based on https://semver.org/#is-there-a-suggested-regular-expression-regex-to-check-a-semver-string
"^(v|release-v)(?P<major>0|[1-9]\\d*)\\.(?P<minor>0|[1-9]\\d*)\\.(?P<patch>0|[1-9]\\d*)(?:-(?P<prerelease>(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+(?P<buildmetadata>[0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$",
}
client = &http.Client{Timeout: 5 * time.Second}
)

// Tag represents a tag in the repository.
Expand All @@ -46,12 +47,11 @@ type TagsResponse struct {

func main() {
repo := flag.String("repo", "kuadrant/kuadrant-operator", "Repository name")
baseURL := flag.String("base-url", "https://quay.io/api/v1/repository/", "Base API URL")
baseURL := flag.String("base-url", "https://quay.io/api/v1/repository", "Base API URL")
dryRun := flag.Bool("dry-run", true, "Dry run")
batchSize := flag.Int("batch-size", 50, "Batch size for deletion. API calls might get rate limited at large values")
flag.Parse()

client := &http.Client{}
logger := log.New(os.Stdout, "INFO: ", log.Ldate|log.Ltime)

if accessToken == "" {
Expand All @@ -62,7 +62,7 @@ func main() {

// Fetch tags from the API
logger.Println("Fetching tags from Quay")
tags, err := fetchTags(client, baseURL, repo)
tags, err := fetchTags(baseURL, repo, accessToken)
if err != nil {
logger.Fatalln("Error fetching tags:", err)
}
Expand Down Expand Up @@ -90,18 +90,18 @@ func main() {
go func(tagName string) {
defer wg.Done()

if dryRun != nil && *dryRun {
if *dryRun {
logger.Printf("DRY RUN - Successfully deleted tag: %s\n", tagName)
} else {
if err := deleteTag(client, baseURL, repo, accessToken, tagName); err != nil {
if err := deleteTag(baseURL, repo, accessToken, tagName); err != nil {
logger.Println(err)
} else {
logger.Printf("Successfully deleted tag: %s\n", tagName)
}

logger.Printf("Successfully deleted tag: %s\n", tagName)
}
}(tagName)

delete(tagsToDelete, tagName) // Remove deleted tag from remainingTags
delete(tagsToDelete, tagName) // Remove deleted tag from tagsToDelete
i++
}

Expand All @@ -115,7 +115,7 @@ func main() {

// fetchTags retrieves the tags from the repository using the Quay.io API.
// https://docs.quay.io/api/swagger/#!/tag/listRepoTags
func fetchTags(client remote.Client, baseURL, repo *string) ([]Tag, error) {
func fetchTags(baseURL, repo *string, accessToken string) ([]Tag, error) {
if baseURL == nil || repo == nil {
return nil, fmt.Errorf("baseURL or repo required")
}
Expand All @@ -124,14 +124,14 @@ func fetchTags(client remote.Client, baseURL, repo *string) ([]Tag, error) {
i := 1

for {
url := fmt.Sprintf("%s%s/tag/?page=%d&limit=%d", *baseURL, *repo, i, pageLimit)
url := fmt.Sprintf("%s/%s/tag/?page=%d&limit=%d", *baseURL, *repo, i, pageLimit)
req, err := http.NewRequest("GET", url, nil)
if err != nil {
return nil, fmt.Errorf("error creating request: %w", err)
}

// Required for private repos
req.Header.Add("Authorization", "Bearer "+accessToken)
req.Header.Add("Authorization", fmt.Sprintf("Bearer %s", accessToken))

// Execute the request
resp, err := client.Do(req)
Expand All @@ -142,7 +142,10 @@ func fetchTags(client remote.Client, baseURL, repo *string) ([]Tag, error) {

// Handle possible non-200 status codes
if resp.StatusCode != http.StatusOK {
body, _ := io.ReadAll(resp.Body)
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, fmt.Errorf("error reading response body: %w", err)
}
return nil, fmt.Errorf("error: received status code %d\nBody: %s", resp.StatusCode, string(body))
}

Expand All @@ -161,7 +164,7 @@ func fetchTags(client remote.Client, baseURL, repo *string) ([]Tag, error) {
allTags = append(allTags, tagsResp.Tags...)

if tagsResp.HasAdditional {
i += 1
i++
continue
}

Expand All @@ -175,40 +178,43 @@ func fetchTags(client remote.Client, baseURL, repo *string) ([]Tag, error) {
// deleteTag sends a DELETE request to remove the specified tag from the repository
// Returns nil if successful, error otherwise
// https://docs.quay.io/api/swagger/#!/tag/deleteFullTag
func deleteTag(client remote.Client, baseURL, repo *string, accessToken, tagName string) error {
func deleteTag(baseURL, repo *string, accessToken, tagName string) error {
if baseURL == nil || repo == nil {
return fmt.Errorf("baseURL or repo required")
}

url := fmt.Sprintf("%s%s/tag/%s", *baseURL, *repo, tagName)
url := fmt.Sprintf("%s/%s/tag/%s", *baseURL, *repo, tagName)

req, err := http.NewRequest("DELETE", url, nil)
if err != nil {
return fmt.Errorf("error creating DELETE request: %s", err)
return fmt.Errorf("error creating DELETE request: %w", err)
}
req.Header.Add("Authorization", "Bearer "+accessToken)
req.Header.Add("Authorization", fmt.Sprintf("Bearer %s", accessToken))

resp, err := client.Do(req)
if err != nil {
return fmt.Errorf("error deleting tag: %s", err)
return fmt.Errorf("error deleting tag: %w", err)
}
defer resp.Body.Close()

if resp.StatusCode == http.StatusNoContent {
return nil
}

body, _ := io.ReadAll(resp.Body)
return fmt.Errorf("Failed to delete tag %s: Status code %d\nBody: %s\n", tagName, resp.StatusCode, string(body))
body, err := io.ReadAll(resp.Body)
if err != nil {
return fmt.Errorf("error reading response body: %w", err)
}
return fmt.Errorf("failed to delete tag %s: Status code %d Body: %s", tagName, resp.StatusCode, string(body))
}

// filterTags takes a slice of tags and preserves string regex and returns two maps: one for tags to delete and one for remaining tags.
// filterTags takes a slice of tags and preserves string regex and returns two maps: one for tags to delete and one for preserved tags.
func filterTags(tags []Tag, preserveSubstrings []string) (map[string]struct{}, map[string]struct{}, error) {
tagsToDelete := make(map[string]struct{})
preservedTags := make(map[string]struct{})

// Compile the regexes for each preserve substring
var preserveRegexes []*regexp.Regexp
preserveRegexes := make([]*regexp.Regexp, 0, len(preserveSubstrings))
for _, substr := range preserveSubstrings {
regex, err := regexp.Compile(substr)
if err != nil {
Expand All @@ -219,7 +225,7 @@ func filterTags(tags []Tag, preserveSubstrings []string) (map[string]struct{}, m

for _, tag := range tags {
// Tags that have an expiration set are ignored as they could be historical tags that have already expired
// i.e. when an existing tag is updated, the previous tag of the same name is expired and is returned when listing
// i.e. when an existing tag is updated, the previous tag of the same name is expired and is still returned when listing
// the tags
if tag.Expiration != "" {
continue
Expand Down
Loading

0 comments on commit e5ea790

Please sign in to comment.