mirror of
				https://github.com/fluxcd/flux2.git
				synced 2025-11-04 03:46:24 +08:00 
			
		
		
		
	* Use `LocalObjectReference` and `NamespacedObjectKindReference` from `meta` package, as required by controller API changes. * Remove `Update` field from created `ImageUpdateAutomation`, as the API changed and the default is now defined in the Custom Resource Definition. Signed-off-by: Hidde Beydals <hello@hidde.co>
		
			
				
	
	
		
			348 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2020 The Flux 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 main
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"crypto/elliptic"
 | 
						|
	"fmt"
 | 
						|
	"io/ioutil"
 | 
						|
	"net/url"
 | 
						|
	"os"
 | 
						|
 | 
						|
	"github.com/manifoldco/promptui"
 | 
						|
	"github.com/spf13/cobra"
 | 
						|
	corev1 "k8s.io/api/core/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/api/errors"
 | 
						|
	apimeta "k8s.io/apimachinery/pkg/api/meta"
 | 
						|
	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/types"
 | 
						|
	"k8s.io/apimachinery/pkg/util/wait"
 | 
						|
	"sigs.k8s.io/controller-runtime/pkg/client"
 | 
						|
 | 
						|
	"github.com/fluxcd/pkg/apis/meta"
 | 
						|
	sourcev1 "github.com/fluxcd/source-controller/api/v1beta1"
 | 
						|
 | 
						|
	"github.com/fluxcd/flux2/internal/flags"
 | 
						|
	"github.com/fluxcd/flux2/internal/utils"
 | 
						|
)
 | 
						|
 | 
						|
type SourceGitFlags struct {
 | 
						|
	GitURL      string
 | 
						|
	GitBranch   string
 | 
						|
	GitTag      string
 | 
						|
	GitSemver   string
 | 
						|
	GitUsername string
 | 
						|
	GitPassword string
 | 
						|
 | 
						|
	GitKeyAlgorithm   flags.PublicKeyAlgorithm
 | 
						|
	GitRSABits        flags.RSAKeyBits
 | 
						|
	GitECDSACurve     flags.ECDSACurve
 | 
						|
	GitSecretRef      string
 | 
						|
	GitImplementation flags.GitImplementation
 | 
						|
}
 | 
						|
 | 
						|
var createSourceGitCmd = &cobra.Command{
 | 
						|
	Use:   "git [name]",
 | 
						|
	Short: "Create or update a GitRepository source",
 | 
						|
	Long: `
 | 
						|
The create source git command generates a GitRepository resource and waits for it to sync.
 | 
						|
For Git over SSH, host and SSH keys are automatically generated and stored in a Kubernetes secret.
 | 
						|
For private Git repositories, the basic authentication credentials are stored in a Kubernetes secret.`,
 | 
						|
	Example: `  # Create a source from a public Git repository master branch
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=https://github.com/stefanprodan/podinfo \
 | 
						|
    --branch=master
 | 
						|
 | 
						|
  # Create a source from a Git repository pinned to specific git tag
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=https://github.com/stefanprodan/podinfo \
 | 
						|
    --tag="3.2.3"
 | 
						|
 | 
						|
  # Create a source from a public Git repository tag that matches a semver range
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=https://github.com/stefanprodan/podinfo \
 | 
						|
    --tag-semver=">=3.2.0 <3.3.0"
 | 
						|
 | 
						|
  # Create a source from a Git repository using SSH authentication
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=ssh://git@github.com/stefanprodan/podinfo \
 | 
						|
    --branch=master
 | 
						|
 | 
						|
  # Create a source from a Git repository using SSH authentication and an
 | 
						|
  # ECDSA P-521 curve public key
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=ssh://git@github.com/stefanprodan/podinfo \
 | 
						|
    --branch=master \
 | 
						|
    --ssh-key-algorithm=ecdsa \
 | 
						|
    --ssh-ecdsa-curve=p521
 | 
						|
 | 
						|
  # Create a source from a Git repository using basic authentication
 | 
						|
  flux create source git podinfo \
 | 
						|
    --url=https://github.com/stefanprodan/podinfo \
 | 
						|
    --username=username \
 | 
						|
    --password=password
 | 
						|
`,
 | 
						|
	RunE: createSourceGitCmdRun,
 | 
						|
}
 | 
						|
 | 
						|
var sourceArgs = NewSourceGitFlags()
 | 
						|
 | 
						|
func init() {
 | 
						|
	createSourceGitCmd.Flags().StringVar(&sourceArgs.GitURL, "url", "", "git address, e.g. ssh://git@host/org/repository")
 | 
						|
	createSourceGitCmd.Flags().StringVar(&sourceArgs.GitBranch, "branch", "master", "git branch")
 | 
						|
	createSourceGitCmd.Flags().StringVar(&sourceArgs.GitTag, "tag", "", "git tag")
 | 
						|
	createSourceGitCmd.Flags().StringVar(&sourceArgs.GitSemver, "tag-semver", "", "git tag semver range")
 | 
						|
	createSourceGitCmd.Flags().StringVarP(&sourceArgs.GitUsername, "username", "u", "", "basic authentication username")
 | 
						|
	createSourceGitCmd.Flags().StringVarP(&sourceArgs.GitPassword, "password", "p", "", "basic authentication password")
 | 
						|
	createSourceGitCmd.Flags().Var(&sourceArgs.GitKeyAlgorithm, "ssh-key-algorithm", sourceArgs.GitKeyAlgorithm.Description())
 | 
						|
	createSourceGitCmd.Flags().Var(&sourceArgs.GitRSABits, "ssh-rsa-bits", sourceArgs.GitRSABits.Description())
 | 
						|
	createSourceGitCmd.Flags().Var(&sourceArgs.GitECDSACurve, "ssh-ecdsa-curve", sourceArgs.GitECDSACurve.Description())
 | 
						|
	createSourceGitCmd.Flags().StringVarP(&sourceArgs.GitSecretRef, "secret-ref", "", "", "the name of an existing secret containing SSH or basic credentials")
 | 
						|
	createSourceGitCmd.Flags().Var(&sourceArgs.GitImplementation, "git-implementation", sourceArgs.GitImplementation.Description())
 | 
						|
 | 
						|
	createSourceCmd.AddCommand(createSourceGitCmd)
 | 
						|
}
 | 
						|
 | 
						|
func NewSourceGitFlags() SourceGitFlags {
 | 
						|
	return SourceGitFlags{
 | 
						|
		GitKeyAlgorithm: "rsa",
 | 
						|
		GitRSABits:      2048,
 | 
						|
		GitECDSACurve:   flags.ECDSACurve{Curve: elliptic.P384()},
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func createSourceGitCmdRun(cmd *cobra.Command, args []string) error {
 | 
						|
	if len(args) < 1 {
 | 
						|
		return fmt.Errorf("GitRepository source name is required")
 | 
						|
	}
 | 
						|
	name := args[0]
 | 
						|
 | 
						|
	if sourceArgs.GitURL == "" {
 | 
						|
		return fmt.Errorf("url is required")
 | 
						|
	}
 | 
						|
 | 
						|
	tmpDir, err := ioutil.TempDir("", name)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer os.RemoveAll(tmpDir)
 | 
						|
 | 
						|
	u, err := url.Parse(sourceArgs.GitURL)
 | 
						|
	if err != nil {
 | 
						|
		return fmt.Errorf("git URL parse failed: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	sourceLabels, err := parseLabels()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	gitRepository := sourcev1.GitRepository{
 | 
						|
		ObjectMeta: metav1.ObjectMeta{
 | 
						|
			Name:      name,
 | 
						|
			Namespace: rootArgs.namespace,
 | 
						|
			Labels:    sourceLabels,
 | 
						|
		},
 | 
						|
		Spec: sourcev1.GitRepositorySpec{
 | 
						|
			URL: sourceArgs.GitURL,
 | 
						|
			Interval: metav1.Duration{
 | 
						|
				Duration: createArgs.interval,
 | 
						|
			},
 | 
						|
			Reference: &sourcev1.GitRepositoryRef{},
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	if sourceArgs.GitImplementation != "" {
 | 
						|
		gitRepository.Spec.GitImplementation = sourceArgs.GitImplementation.String()
 | 
						|
	}
 | 
						|
 | 
						|
	if sourceArgs.GitSemver != "" {
 | 
						|
		gitRepository.Spec.Reference.SemVer = sourceArgs.GitSemver
 | 
						|
	} else if sourceArgs.GitTag != "" {
 | 
						|
		gitRepository.Spec.Reference.Tag = sourceArgs.GitTag
 | 
						|
	} else {
 | 
						|
		gitRepository.Spec.Reference.Branch = sourceArgs.GitBranch
 | 
						|
	}
 | 
						|
 | 
						|
	if createArgs.export {
 | 
						|
		if sourceArgs.GitSecretRef != "" {
 | 
						|
			gitRepository.Spec.SecretRef = &meta.LocalObjectReference{
 | 
						|
				Name: sourceArgs.GitSecretRef,
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return exportGit(gitRepository)
 | 
						|
	}
 | 
						|
 | 
						|
	ctx, cancel := context.WithTimeout(context.Background(), rootArgs.timeout)
 | 
						|
	defer cancel()
 | 
						|
 | 
						|
	kubeClient, err := utils.KubeClient(rootArgs.kubeconfig, rootArgs.kubecontext)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	withAuth := false
 | 
						|
	// TODO(hidde): move all auth prep to separate func?
 | 
						|
	if sourceArgs.GitSecretRef != "" {
 | 
						|
		withAuth = true
 | 
						|
	} else if u.Scheme == "ssh" {
 | 
						|
		logger.Generatef("generating deploy key pair")
 | 
						|
		pair, err := generateKeyPair(ctx, sourceArgs.GitKeyAlgorithm, sourceArgs.GitRSABits, sourceArgs.GitECDSACurve)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		logger.Successf("deploy key: %s", pair.PublicKey)
 | 
						|
		prompt := promptui.Prompt{
 | 
						|
			Label:     "Have you added the deploy key to your repository",
 | 
						|
			IsConfirm: true,
 | 
						|
		}
 | 
						|
		if _, err := prompt.Run(); err != nil {
 | 
						|
			return fmt.Errorf("aborting")
 | 
						|
		}
 | 
						|
 | 
						|
		logger.Actionf("collecting preferred public key from SSH server")
 | 
						|
		hostKey, err := scanHostKey(ctx, u)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		logger.Successf("collected public key from SSH server:\n%s", hostKey)
 | 
						|
 | 
						|
		logger.Actionf("applying secret with keys")
 | 
						|
		secret := corev1.Secret{
 | 
						|
			ObjectMeta: metav1.ObjectMeta{
 | 
						|
				Name:      name,
 | 
						|
				Namespace: rootArgs.namespace,
 | 
						|
				Labels:    sourceLabels,
 | 
						|
			},
 | 
						|
			StringData: map[string]string{
 | 
						|
				"identity":     string(pair.PrivateKey),
 | 
						|
				"identity.pub": string(pair.PublicKey),
 | 
						|
				"known_hosts":  string(hostKey),
 | 
						|
			},
 | 
						|
		}
 | 
						|
		if err := upsertSecret(ctx, kubeClient, secret); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		withAuth = true
 | 
						|
	} else if sourceArgs.GitUsername != "" && sourceArgs.GitPassword != "" {
 | 
						|
		logger.Actionf("applying secret with basic auth credentials")
 | 
						|
		secret := corev1.Secret{
 | 
						|
			ObjectMeta: metav1.ObjectMeta{
 | 
						|
				Name:      name,
 | 
						|
				Namespace: rootArgs.namespace,
 | 
						|
				Labels:    sourceLabels,
 | 
						|
			},
 | 
						|
			StringData: map[string]string{
 | 
						|
				"username": sourceArgs.GitUsername,
 | 
						|
				"password": sourceArgs.GitPassword,
 | 
						|
			},
 | 
						|
		}
 | 
						|
		if err := upsertSecret(ctx, kubeClient, secret); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		withAuth = true
 | 
						|
	}
 | 
						|
 | 
						|
	if withAuth {
 | 
						|
		logger.Successf("authentication configured")
 | 
						|
	}
 | 
						|
 | 
						|
	logger.Generatef("generating GitRepository source")
 | 
						|
 | 
						|
	if withAuth {
 | 
						|
		secretName := name
 | 
						|
		if sourceArgs.GitSecretRef != "" {
 | 
						|
			secretName = sourceArgs.GitSecretRef
 | 
						|
		}
 | 
						|
		gitRepository.Spec.SecretRef = &meta.LocalObjectReference{
 | 
						|
			Name: secretName,
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	logger.Actionf("applying GitRepository source")
 | 
						|
	namespacedName, err := upsertGitRepository(ctx, kubeClient, &gitRepository)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	logger.Waitingf("waiting for GitRepository source reconciliation")
 | 
						|
	if err := wait.PollImmediate(rootArgs.pollInterval, rootArgs.timeout,
 | 
						|
		isGitRepositoryReady(ctx, kubeClient, namespacedName, &gitRepository)); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	logger.Successf("GitRepository source reconciliation completed")
 | 
						|
 | 
						|
	if gitRepository.Status.Artifact == nil {
 | 
						|
		return fmt.Errorf("GitRepository source reconciliation completed but no artifact was found")
 | 
						|
	}
 | 
						|
	logger.Successf("fetched revision: %s", gitRepository.Status.Artifact.Revision)
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func upsertGitRepository(ctx context.Context, kubeClient client.Client,
 | 
						|
	gitRepository *sourcev1.GitRepository) (types.NamespacedName, error) {
 | 
						|
	namespacedName := types.NamespacedName{
 | 
						|
		Namespace: gitRepository.GetNamespace(),
 | 
						|
		Name:      gitRepository.GetName(),
 | 
						|
	}
 | 
						|
 | 
						|
	var existing sourcev1.GitRepository
 | 
						|
	err := kubeClient.Get(ctx, namespacedName, &existing)
 | 
						|
	if err != nil {
 | 
						|
		if errors.IsNotFound(err) {
 | 
						|
			if err := kubeClient.Create(ctx, gitRepository); err != nil {
 | 
						|
				return namespacedName, err
 | 
						|
			} else {
 | 
						|
				logger.Successf("GitRepository source created")
 | 
						|
				return namespacedName, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return namespacedName, err
 | 
						|
	}
 | 
						|
 | 
						|
	existing.Labels = gitRepository.Labels
 | 
						|
	existing.Spec = gitRepository.Spec
 | 
						|
	if err := kubeClient.Update(ctx, &existing); err != nil {
 | 
						|
		return namespacedName, err
 | 
						|
	}
 | 
						|
	gitRepository = &existing
 | 
						|
	logger.Successf("GitRepository source updated")
 | 
						|
	return namespacedName, nil
 | 
						|
}
 | 
						|
 | 
						|
func isGitRepositoryReady(ctx context.Context, kubeClient client.Client,
 | 
						|
	namespacedName types.NamespacedName, gitRepository *sourcev1.GitRepository) wait.ConditionFunc {
 | 
						|
	return func() (bool, error) {
 | 
						|
		err := kubeClient.Get(ctx, namespacedName, gitRepository)
 | 
						|
		if err != nil {
 | 
						|
			return false, err
 | 
						|
		}
 | 
						|
 | 
						|
		if c := apimeta.FindStatusCondition(gitRepository.Status.Conditions, meta.ReadyCondition); c != nil {
 | 
						|
			switch c.Status {
 | 
						|
			case metav1.ConditionTrue:
 | 
						|
				return true, nil
 | 
						|
			case metav1.ConditionFalse:
 | 
						|
				return false, fmt.Errorf(c.Message)
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return false, nil
 | 
						|
	}
 | 
						|
}
 |