mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-05-17 18:25:26 +00:00
* initial changes to rename OrgRepo to RepoPath * changes to rename Path to KustRootPath * addressed review comments * addressed review comments * docs: Add documentation for namespace transformer Add a short description of the namespace transformer and example usage to examples/transformerconfigs/README.md. References: #629 Signed-off-by: Lars Kellogg-Stedman <lars@oddbit.com> * Localize patchesJson6902, patchesStrategicMerge, replacements (#4904) * Localize patchesJson6902, patchesStrategicMerge, replacements * Address code review feedback * Improve readability * Remove deprecation warning check * Load legacy kustomization fields for `localize` (#4918) * Load legacy kustomization * Expose loadKustFile in kusttarget * remove FixKustomizationPreUnmarshalling * remove deprecated cfg and fn commands (#4930) * remove deprecated cfg and fn commands * fix lint error * run gofmt * Localize PatchTransformer, PatchJson6902Transformer (#4920) * Localize patches, patchesJson6902 custom transformers * Improve readability * Localize fields: openapi, configurations, crds (#4907) * Localize openapi, configurations, crds * Add integration test * Move krusty test * Address code review feedback * Implement locRootPath (#4909) * Implement locRootPath, and include userinfo, port in locFilePath * Strip userinfo, port * Improve readability * Localize legacy fields * Localize resources (#4912) * Localize resources * Improve readability * Add integration tests * Group test helper functions * Remove Functionality that Pulls Env Variables from Empty Keys * Update api/kv/kv.go Co-authored-by: Katrina Verey <kn.verey@gmail.com> * refactor Unmarshal Kustomization struct code * improve error messages * Run go mod tidy on all modules before update * Update sigs.k8s.io/yaml to 1.3.0 * fixed test failure because of latest commits Signed-off-by: Lars Kellogg-Stedman <lars@oddbit.com> Co-authored-by: Lars Kellogg-Stedman <lars@oddbit.com> Co-authored-by: Anna Song <annasong@google.com> Co-authored-by: yugo kobayashi <kobdotsh@gmail.com> Co-authored-by: Natasha Sarkar <natashasarkar@google.com> Co-authored-by: Cailyn Edwards <cailyn.edwards@shopify.com> Co-authored-by: Cailyn <cailyn.s.e@gmail.com> Co-authored-by: Katrina Verey <kn.verey@gmail.com> Co-authored-by: Katrina Verey <katrina.verey@shopify.com>
300 lines
8.2 KiB
Go
300 lines
8.2 KiB
Go
// Copyright 2019 The Kubernetes Authors.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package git
|
|
|
|
import (
|
|
"fmt"
|
|
"net/url"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"sigs.k8s.io/kustomize/kyaml/filesys"
|
|
)
|
|
|
|
// Used as a temporary non-empty occupant of the cloneDir
|
|
// field, as something distinguishable from the empty string
|
|
// in various outputs (especially tests). Not using an
|
|
// actual directory name here, as that's a temporary directory
|
|
// with a unique name that isn't created until clone time.
|
|
const notCloned = filesys.ConfirmedDir("/notCloned")
|
|
|
|
// RepoSpec specifies a git repository and a branch and path therein.
|
|
type RepoSpec struct {
|
|
// Raw, original spec, used to look for cycles.
|
|
// TODO(monopole): Drop raw, use processed fields instead.
|
|
raw string
|
|
|
|
// Host, e.g. https://github.com/
|
|
Host string
|
|
|
|
// RepoPath name (Path to repository),
|
|
// e.g. kubernetes-sigs/kustomize
|
|
RepoPath string
|
|
|
|
// Dir is where the repository is cloned to.
|
|
Dir filesys.ConfirmedDir
|
|
|
|
// Relative path in the repository, and in the cloneDir,
|
|
// to a Kustomization.
|
|
KustRootPath string
|
|
|
|
// Branch or tag reference.
|
|
Ref string
|
|
|
|
// e.g. .git or empty in case of _git is present
|
|
GitSuffix string
|
|
|
|
// Submodules indicates whether or not to clone git submodules.
|
|
Submodules bool
|
|
|
|
// Timeout is the maximum duration allowed for execing git commands.
|
|
Timeout time.Duration
|
|
}
|
|
|
|
// CloneSpec returns a string suitable for "git clone {spec}".
|
|
func (x *RepoSpec) CloneSpec() string {
|
|
if isAzureHost(x.Host) || isAWSHost(x.Host) {
|
|
return x.Host + x.RepoPath
|
|
}
|
|
return x.Host + x.RepoPath + x.GitSuffix
|
|
}
|
|
|
|
func (x *RepoSpec) CloneDir() filesys.ConfirmedDir {
|
|
return x.Dir
|
|
}
|
|
|
|
func (x *RepoSpec) Raw() string {
|
|
return x.raw
|
|
}
|
|
|
|
func (x *RepoSpec) AbsPath() string {
|
|
return x.Dir.Join(x.KustRootPath)
|
|
}
|
|
|
|
func (x *RepoSpec) Cleaner(fSys filesys.FileSystem) func() error {
|
|
return func() error { return fSys.RemoveAll(x.Dir.String()) }
|
|
}
|
|
|
|
// NewRepoSpecFromURL parses git-like urls.
|
|
// From strings like git@github.com:someOrg/someRepo.git or
|
|
// https://github.com/someOrg/someRepo?ref=someHash, extract
|
|
// the parts.
|
|
func NewRepoSpecFromURL(n string) (*RepoSpec, error) {
|
|
if filepath.IsAbs(n) {
|
|
return nil, fmt.Errorf("uri looks like abs path: %s", n)
|
|
}
|
|
repoSpecVal := parseGitURL(n)
|
|
if repoSpecVal.RepoPath == "" {
|
|
return nil, fmt.Errorf("url lacks repoPath: %s", n)
|
|
}
|
|
if repoSpecVal.Host == "" {
|
|
return nil, fmt.Errorf("url lacks host: %s", n)
|
|
}
|
|
cleanedPath := filepath.Clean(strings.TrimPrefix(repoSpecVal.KustRootPath, string(filepath.Separator)))
|
|
if pathElements := strings.Split(cleanedPath, string(filepath.Separator)); len(pathElements) > 0 &&
|
|
pathElements[0] == filesys.ParentDir {
|
|
return nil, fmt.Errorf("url path exits repo: %s", n)
|
|
}
|
|
return repoSpecVal, nil
|
|
}
|
|
|
|
const (
|
|
refQuery = "?ref="
|
|
gitSuffix = ".git"
|
|
gitDelimiter = "_git/"
|
|
)
|
|
|
|
// From strings like git@github.com:someOrg/someRepo.git or
|
|
// https://github.com/someOrg/someRepo?ref=someHash, extract
|
|
// the different parts of URL , set into a RepoSpec object and return RepoSpec object.
|
|
func parseGitURL(n string) *RepoSpec {
|
|
repoSpec := &RepoSpec{raw: n, Dir: notCloned, Timeout: defaultTimeout, Submodules: defaultSubmodules}
|
|
// parse query first
|
|
// safe because according to rfc3986: ? only allowed in query
|
|
// and not recognized %-encoded
|
|
beforeQuery, query, _ := strings.Cut(n, "?")
|
|
n = beforeQuery
|
|
// if no query, defaults returned
|
|
repoSpec.Ref, repoSpec.Timeout, repoSpec.Submodules = parseQuery(query)
|
|
|
|
if strings.Contains(n, gitDelimiter) {
|
|
index := strings.Index(n, gitDelimiter)
|
|
// Adding _git/ to host
|
|
repoSpec.Host = normalizeGitHostSpec(n[:index+len(gitDelimiter)])
|
|
repoSpec.RepoPath = strings.Split(n[index+len(gitDelimiter):], "/")[0]
|
|
repoSpec.KustRootPath = parsePath(n[index+len(gitDelimiter)+len(repoSpec.RepoPath):])
|
|
return repoSpec
|
|
}
|
|
repoSpec.Host, n = parseHostSpec(n)
|
|
isLocal := strings.HasPrefix(repoSpec.Host, "file://")
|
|
if !isLocal {
|
|
repoSpec.GitSuffix = gitSuffix
|
|
}
|
|
if strings.Contains(n, gitSuffix) {
|
|
repoSpec.GitSuffix = gitSuffix
|
|
index := strings.Index(n, gitSuffix)
|
|
repoSpec.RepoPath = n[0:index]
|
|
n = n[index+len(gitSuffix):]
|
|
if len(n) > 0 && n[0] == '/' {
|
|
n = n[1:]
|
|
}
|
|
repoSpec.KustRootPath = parsePath(n)
|
|
return repoSpec
|
|
}
|
|
|
|
if isLocal {
|
|
if idx := strings.Index(n, "//"); idx > 0 {
|
|
repoSpec.RepoPath = n[:idx]
|
|
n = n[idx+2:]
|
|
repoSpec.KustRootPath = parsePath(n)
|
|
return repoSpec
|
|
}
|
|
repoSpec.RepoPath = parsePath(n)
|
|
return repoSpec
|
|
}
|
|
|
|
i := strings.Index(n, "/")
|
|
if i < 1 {
|
|
repoSpec.KustRootPath = parsePath(n)
|
|
return repoSpec
|
|
}
|
|
j := strings.Index(n[i+1:], "/")
|
|
if j >= 0 {
|
|
j += i + 1
|
|
repoSpec.RepoPath = n[:j]
|
|
repoSpec.KustRootPath = parsePath(n[j+1:])
|
|
return repoSpec
|
|
}
|
|
repoSpec.KustRootPath = ""
|
|
repoSpec.RepoPath = parsePath(n)
|
|
return repoSpec
|
|
}
|
|
|
|
// Clone git submodules by default.
|
|
const defaultSubmodules = true
|
|
|
|
// Arbitrary, but non-infinite, timeout for running commands.
|
|
const defaultTimeout = 27 * time.Second
|
|
|
|
func parseQuery(query string) (string, time.Duration, bool) {
|
|
values, err := url.ParseQuery(query)
|
|
// in event of parse failure, return defaults
|
|
if err != nil {
|
|
return "", defaultTimeout, defaultSubmodules
|
|
}
|
|
|
|
// ref is the desired git ref to target. Can be specified by in a git URL
|
|
// with ?ref=<string> or ?version=<string>, although ref takes precedence.
|
|
ref := values.Get("version")
|
|
if queryValue := values.Get("ref"); queryValue != "" {
|
|
ref = queryValue
|
|
}
|
|
|
|
// depth is the desired git exec timeout. Can be specified by in a git URL
|
|
// with ?timeout=<duration>.
|
|
duration := defaultTimeout
|
|
if queryValue := values.Get("timeout"); queryValue != "" {
|
|
// Attempt to first parse as a number of integer seconds (like "61"),
|
|
// and then attempt to parse as a suffixed duration (like "61s").
|
|
if intValue, err := strconv.Atoi(queryValue); err == nil && intValue > 0 {
|
|
duration = time.Duration(intValue) * time.Second
|
|
} else if durationValue, err := time.ParseDuration(queryValue); err == nil && durationValue > 0 {
|
|
duration = durationValue
|
|
}
|
|
}
|
|
|
|
// submodules indicates if git submodule cloning is desired. Can be
|
|
// specified by in a git URL with ?submodules=<bool>.
|
|
submodules := defaultSubmodules
|
|
if queryValue := values.Get("submodules"); queryValue != "" {
|
|
if boolValue, err := strconv.ParseBool(queryValue); err == nil {
|
|
submodules = boolValue
|
|
}
|
|
}
|
|
|
|
return ref, duration, submodules
|
|
}
|
|
|
|
func parsePath(n string) string {
|
|
parsed, err := url.Parse(n)
|
|
// TODO(annasong): decide how to handle error, i.e. return error, empty string, etc.
|
|
if err != nil {
|
|
return n
|
|
}
|
|
return parsed.Path
|
|
}
|
|
|
|
func parseHostSpec(n string) (string, string) {
|
|
var host string
|
|
// Start accumulating the host part.
|
|
for _, p := range []string{
|
|
// Order matters here.
|
|
"git::", "gh:", "ssh://", "https://", "http://", "file://",
|
|
"git@", "github.com:", "github.com/"} {
|
|
if len(p) < len(n) && strings.ToLower(n[:len(p)]) == p {
|
|
n = n[len(p):]
|
|
host += p
|
|
}
|
|
}
|
|
if host == "git@" {
|
|
i := strings.Index(n, "/")
|
|
if i > -1 {
|
|
host += n[:i+1]
|
|
n = n[i+1:]
|
|
} else {
|
|
i = strings.Index(n, ":")
|
|
if i > -1 {
|
|
host += n[:i+1]
|
|
n = n[i+1:]
|
|
}
|
|
}
|
|
return host, n
|
|
}
|
|
|
|
// If host is a http(s) or ssh URL, grab the domain part.
|
|
for _, p := range []string{
|
|
"ssh://", "https://", "http://"} {
|
|
if strings.HasSuffix(host, p) {
|
|
i := strings.Index(n, "/")
|
|
if i > -1 {
|
|
host += n[0 : i+1]
|
|
n = n[i+1:]
|
|
}
|
|
break
|
|
}
|
|
}
|
|
|
|
return normalizeGitHostSpec(host), n
|
|
}
|
|
|
|
func normalizeGitHostSpec(host string) string {
|
|
s := strings.ToLower(host)
|
|
if strings.Contains(s, "github.com") {
|
|
if strings.Contains(s, "git@") || strings.Contains(s, "ssh:") {
|
|
host = "git@github.com:"
|
|
} else {
|
|
host = "https://github.com/"
|
|
}
|
|
}
|
|
if strings.HasPrefix(s, "git::") {
|
|
host = strings.TrimPrefix(s, "git::")
|
|
}
|
|
return host
|
|
}
|
|
|
|
// The format of Azure repo URL is documented
|
|
// https://docs.microsoft.com/en-us/azure/devops/repos/git/clone?view=vsts&tabs=visual-studio#clone_url
|
|
func isAzureHost(host string) bool {
|
|
return strings.Contains(host, "dev.azure.com") ||
|
|
strings.Contains(host, "visualstudio.com")
|
|
}
|
|
|
|
// The format of AWS repo URL is documented
|
|
// https://docs.aws.amazon.com/codecommit/latest/userguide/regions.html
|
|
func isAWSHost(host string) bool {
|
|
return strings.Contains(host, "amazonaws.com")
|
|
}
|