mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-06-29 09:40:49 +00:00
Compare commits
1 Commits
release-ap
...
kustomize/
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5ba90fe5ef |
@@ -1,8 +0,0 @@
|
|||||||
.github
|
|
||||||
docs
|
|
||||||
examples
|
|
||||||
functions
|
|
||||||
hack
|
|
||||||
site
|
|
||||||
travis
|
|
||||||
*.md
|
|
||||||
68
.github/ISSUE_TEMPLATE/bug_report.md
vendored
68
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -1,68 +0,0 @@
|
|||||||
---
|
|
||||||
name: Bug report
|
|
||||||
about: Create a report to help us improve
|
|
||||||
title: ""
|
|
||||||
labels:
|
|
||||||
- kind/bug
|
|
||||||
assignees: ""
|
|
||||||
---
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Please read this page: https://kubernetes-sigs.github.io/kustomize/contributing/bugs/ before
|
|
||||||
filing a bug
|
|
||||||
-->
|
|
||||||
|
|
||||||
<!-- Feel free to skip the sections if they are not applicable. -->
|
|
||||||
|
|
||||||
**Describe the bug**
|
|
||||||
|
|
||||||
<!-- A clear and concise description of what the bug is. -->
|
|
||||||
|
|
||||||
**Files that can reproduce the issue**
|
|
||||||
|
|
||||||
<!--
|
|
||||||
We cannot figure out or fix the issue if we don't know how to reproduce. Please
|
|
||||||
provide a minimum set of files that can reproduce the issue. You can paste the
|
|
||||||
file contents here or provide a link to a tarball or git repo.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
kustomization.yaml
|
|
||||||
|
|
||||||
```
|
|
||||||
apiVersion: kustomize.config.k8s.io/v1beta1
|
|
||||||
kind: Kustomization
|
|
||||||
resources:
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
resources.yaml
|
|
||||||
|
|
||||||
```
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
...
|
|
||||||
-->
|
|
||||||
|
|
||||||
**Expected output**
|
|
||||||
|
|
||||||
<!-- What's the expected output? -->
|
|
||||||
|
|
||||||
**Actual output**
|
|
||||||
|
|
||||||
<!-- What's the actual output? -->
|
|
||||||
|
|
||||||
**Kustomize version**
|
|
||||||
|
|
||||||
<!-- Please use the latest version when it's possible. -->
|
|
||||||
|
|
||||||
**Platform**
|
|
||||||
|
|
||||||
<!-- Linux/macOS/Windows? -->
|
|
||||||
|
|
||||||
**Additional context**
|
|
||||||
|
|
||||||
<!-- Add any other context about the problem here. -->
|
|
||||||
1
.github/ISSUE_TEMPLATE/config.yaml
vendored
1
.github/ISSUE_TEMPLATE/config.yaml
vendored
@@ -1 +0,0 @@
|
|||||||
blank_issues_enabled: true
|
|
||||||
26
.github/ISSUE_TEMPLATE/feature_request.md
vendored
26
.github/ISSUE_TEMPLATE/feature_request.md
vendored
@@ -1,26 +0,0 @@
|
|||||||
---
|
|
||||||
name: Feature request
|
|
||||||
about: Suggest an idea for this project
|
|
||||||
title: ""
|
|
||||||
labels:
|
|
||||||
- kind/feature
|
|
||||||
assignees: ""
|
|
||||||
---
|
|
||||||
|
|
||||||
<!-- Feel free to skip the sections if they are not applicable. -->
|
|
||||||
|
|
||||||
**Is your feature request related to a problem? Please describe.**
|
|
||||||
|
|
||||||
<!-- A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] -->
|
|
||||||
|
|
||||||
**Describe the solution you'd like**
|
|
||||||
|
|
||||||
<!-- A clear and concise description of what you want to happen. -->
|
|
||||||
|
|
||||||
**Describe alternatives you've considered**
|
|
||||||
|
|
||||||
<!-- A clear and concise description of any alternative solutions or features you've considered. -->
|
|
||||||
|
|
||||||
**Additional context**
|
|
||||||
|
|
||||||
<!-- Add any other context or screenshots about the feature request here. -->
|
|
||||||
9
.github/ISSUE_TEMPLATE/question.md
vendored
9
.github/ISSUE_TEMPLATE/question.md
vendored
@@ -1,9 +0,0 @@
|
|||||||
---
|
|
||||||
name: Question
|
|
||||||
about: Ask a question about the kustomize
|
|
||||||
title: "[Question]"
|
|
||||||
labels: ""
|
|
||||||
assignees: ""
|
|
||||||
---
|
|
||||||
|
|
||||||
<!-- Please describe your question here -->
|
|
||||||
100
.github/workflows/go.yml
vendored
100
.github/workflows/go.yml
vendored
@@ -1,100 +0,0 @@
|
|||||||
name: Go
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [ master ]
|
|
||||||
pull_request:
|
|
||||||
branches: [ master ]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
|
|
||||||
lint:
|
|
||||||
name: Lint
|
|
||||||
runs-on: [ubuntu-latest]
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: ^1.13
|
|
||||||
id: go
|
|
||||||
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Lint
|
|
||||||
run: ./scripts/kyaml-pre-commit.sh
|
|
||||||
env:
|
|
||||||
KUSTOMIZE_DOCKER_E2E: false # don't need to do e2e tests for linting
|
|
||||||
|
|
||||||
test-linux:
|
|
||||||
name: Test Linux
|
|
||||||
runs-on: [ubuntu-latest]
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: ^1.13
|
|
||||||
id: go
|
|
||||||
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Test kyaml
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./kyaml
|
|
||||||
|
|
||||||
- name: Test cmd/config
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./cmd/config
|
|
||||||
env:
|
|
||||||
KUSTOMIZE_DOCKER_E2E: true
|
|
||||||
|
|
||||||
test-macos:
|
|
||||||
name: Test MacOS
|
|
||||||
runs-on: [macos-latest]
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: ^1.13
|
|
||||||
id: go
|
|
||||||
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Test kyaml
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./kyaml
|
|
||||||
|
|
||||||
- name: Test cmd/config
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./cmd/config
|
|
||||||
env:
|
|
||||||
KUSTOMIZE_DOCKER_E2E: false # docker not installed on mac
|
|
||||||
|
|
||||||
test-windows:
|
|
||||||
name: Test Windows
|
|
||||||
runs-on: [windows-latest]
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Set up Go 1.x
|
|
||||||
uses: actions/setup-go@v2
|
|
||||||
with:
|
|
||||||
go-version: ^1.13
|
|
||||||
id: go
|
|
||||||
|
|
||||||
- name: Check out code into the Go module directory
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Test kyaml
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./kyaml
|
|
||||||
|
|
||||||
- name: Test cmd/config
|
|
||||||
run: go test -cover ./...
|
|
||||||
working-directory: ./cmd/config
|
|
||||||
env:
|
|
||||||
KUSTOMIZE_DOCKER_E2E: false # docker on windows not working well yet
|
|
||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -19,5 +19,3 @@
|
|||||||
|
|
||||||
# macOS
|
# macOS
|
||||||
*.DS_store
|
*.DS_store
|
||||||
|
|
||||||
.bin
|
|
||||||
|
|||||||
@@ -7,13 +7,8 @@ linters:
|
|||||||
- bodyclose
|
- bodyclose
|
||||||
- deadcode
|
- deadcode
|
||||||
- depguard
|
- depguard
|
||||||
# - dogsled
|
|
||||||
- dupl
|
- dupl
|
||||||
# - errcheck
|
|
||||||
# - funlen
|
|
||||||
# - gochecknoinits
|
|
||||||
- goconst
|
- goconst
|
||||||
# - gocritic
|
|
||||||
- gocyclo
|
- gocyclo
|
||||||
- gofmt
|
- gofmt
|
||||||
- goimports
|
- goimports
|
||||||
@@ -26,7 +21,6 @@ linters:
|
|||||||
- lll
|
- lll
|
||||||
- misspell
|
- misspell
|
||||||
- nakedret
|
- nakedret
|
||||||
# - scopelint
|
|
||||||
- staticcheck
|
- staticcheck
|
||||||
- structcheck
|
- structcheck
|
||||||
# stylecheck demands that acronyms not be treated as words
|
# stylecheck demands that acronyms not be treated as words
|
||||||
@@ -34,10 +28,9 @@ linters:
|
|||||||
# - stylecheck
|
# - stylecheck
|
||||||
- typecheck
|
- typecheck
|
||||||
- unconvert
|
- unconvert
|
||||||
- unparam
|
|
||||||
- unused
|
- unused
|
||||||
|
- unparam
|
||||||
- varcheck
|
- varcheck
|
||||||
# - whitespace
|
|
||||||
|
|
||||||
linters-settings:
|
linters-settings:
|
||||||
dupl:
|
dupl:
|
||||||
|
|||||||
42
.travis.yml
Normal file
42
.travis.yml
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
os:
|
||||||
|
- linux
|
||||||
|
- osx
|
||||||
|
# TODO: Speed up the slowness of the osx travis runs
|
||||||
|
# Maybe cache brew installs?
|
||||||
|
#
|
||||||
|
# TODO: Uncomment when some gets the tests running on Windows.
|
||||||
|
# - windows
|
||||||
|
|
||||||
|
addons:
|
||||||
|
apt:
|
||||||
|
packages:
|
||||||
|
- tree
|
||||||
|
homebrew:
|
||||||
|
packages:
|
||||||
|
- tree
|
||||||
|
update: true
|
||||||
|
|
||||||
|
# Only clone the most recent commit.
|
||||||
|
git:
|
||||||
|
depth: 1
|
||||||
|
|
||||||
|
language: go
|
||||||
|
|
||||||
|
go:
|
||||||
|
- "1.13"
|
||||||
|
|
||||||
|
go_import_path: sigs.k8s.io/kustomize
|
||||||
|
|
||||||
|
before_install:
|
||||||
|
- source ./travis/consider-early-travis-exit.sh
|
||||||
|
|
||||||
|
# Skip the install process; let pre-commit.sh do it.
|
||||||
|
install: true
|
||||||
|
|
||||||
|
script:
|
||||||
|
- make verify-kustomize
|
||||||
|
- ./travis/kyaml-pre-commit.sh
|
||||||
|
|
||||||
|
# TBD. Suppressing for now.
|
||||||
|
notifications:
|
||||||
|
email: false
|
||||||
229
Makefile
229
Makefile
@@ -4,19 +4,8 @@
|
|||||||
# Makefile for kustomize CLI and API.
|
# Makefile for kustomize CLI and API.
|
||||||
|
|
||||||
MYGOBIN := $(shell go env GOPATH)/bin
|
MYGOBIN := $(shell go env GOPATH)/bin
|
||||||
SHELL := /usr/bin/env bash
|
SHELL := /bin/bash
|
||||||
export PATH := $(MYGOBIN):$(PATH)
|
export PATH := $(MYGOBIN):$(PATH)
|
||||||
MODULES := '"cmd/config" "api/" "kustomize/" "kyaml/"'
|
|
||||||
|
|
||||||
# Provide defaults for REPO_OWNER and REPO_NAME if not present.
|
|
||||||
# Typically these values would be provided by Prow.
|
|
||||||
ifndef REPO_OWNER
|
|
||||||
REPO_OWNER := "kubernetes-sigs"
|
|
||||||
endif
|
|
||||||
|
|
||||||
ifndef REPO_NAME
|
|
||||||
REPO_NAME := "kustomize"
|
|
||||||
endif
|
|
||||||
|
|
||||||
.PHONY: all
|
.PHONY: all
|
||||||
all: verify-kustomize
|
all: verify-kustomize
|
||||||
@@ -26,24 +15,7 @@ verify-kustomize: \
|
|||||||
lint-kustomize \
|
lint-kustomize \
|
||||||
test-unit-kustomize-all \
|
test-unit-kustomize-all \
|
||||||
test-examples-kustomize-against-HEAD \
|
test-examples-kustomize-against-HEAD \
|
||||||
test-examples-kustomize-against-3.9 \
|
test-examples-kustomize-against-latest
|
||||||
test-examples-kustomize-against-3.8
|
|
||||||
|
|
||||||
# The following target referenced by a file in
|
|
||||||
# https://github.com/kubernetes/test-infra/tree/master/config/jobs/kubernetes-sigs/kustomize
|
|
||||||
.PHONY: prow-presubmit-check
|
|
||||||
prow-presubmit-check: \
|
|
||||||
lint-kustomize \
|
|
||||||
test-multi-module \
|
|
||||||
test-unit-kustomize-all \
|
|
||||||
test-unit-cmd-all \
|
|
||||||
test-go-mod \
|
|
||||||
test-examples-kustomize-against-HEAD \
|
|
||||||
test-examples-kustomize-against-3.9 \
|
|
||||||
test-examples-kustomize-against-3.8
|
|
||||||
|
|
||||||
.PHONY: verify-kustomize-e2e
|
|
||||||
verify-kustomize-e2e: test-examples-e2e-kustomize
|
|
||||||
|
|
||||||
# Other builds in this repo might want a different linter version.
|
# Other builds in this repo might want a different linter version.
|
||||||
# Without one Makefile to rule them all, the different makes
|
# Without one Makefile to rule them all, the different makes
|
||||||
@@ -51,50 +23,39 @@ verify-kustomize-e2e: test-examples-e2e-kustomize
|
|||||||
# since everything uses the same implicit GOPATH.
|
# since everything uses the same implicit GOPATH.
|
||||||
# This installs in a temp dir to avoid overwriting someone else's
|
# This installs in a temp dir to avoid overwriting someone else's
|
||||||
# linter, then installs in MYGOBIN with a new name.
|
# linter, then installs in MYGOBIN with a new name.
|
||||||
# Version pinned by hack/go.mod
|
# Version pinned by api/go.mod
|
||||||
$(MYGOBIN)/golangci-lint-kustomize:
|
$(MYGOBIN)/golangci-lint-kustomize:
|
||||||
( \
|
( \
|
||||||
set -e; \
|
set -e; \
|
||||||
cd hack; \
|
export GOBIN=$$(mktemp -d) \
|
||||||
GO111MODULE=on go build -tags=tools -o $(MYGOBIN)/golangci-lint-kustomize github.com/golangci/golangci-lint/cmd/golangci-lint; \
|
cd api; \
|
||||||
|
GO111MODULE=on go install github.com/golangci/golangci-lint/cmd/golangci-lint; \
|
||||||
|
mv $$GOBIN/golangci-lint $(MYGOBIN)/golangci-lint-kustomize \
|
||||||
)
|
)
|
||||||
|
|
||||||
# Install from version specified in api/go.mod.
|
# Version pinned by api/go.mod
|
||||||
$(MYGOBIN)/mdrip:
|
$(MYGOBIN)/mdrip:
|
||||||
cd api; \
|
cd api; \
|
||||||
go install github.com/monopole/mdrip
|
go install github.com/monopole/mdrip
|
||||||
|
|
||||||
# Install from version specified in api/go.mod.
|
# Version pinned by api/go.mod
|
||||||
$(MYGOBIN)/stringer:
|
$(MYGOBIN)/stringer:
|
||||||
cd api; \
|
cd api; \
|
||||||
go install golang.org/x/tools/cmd/stringer
|
go install golang.org/x/tools/cmd/stringer
|
||||||
|
|
||||||
# Install from version specified in api/go.mod.
|
# Version pinned by api/go.mod
|
||||||
$(MYGOBIN)/goimports:
|
$(MYGOBIN)/goimports:
|
||||||
cd api; \
|
cd api; \
|
||||||
go install golang.org/x/tools/cmd/goimports
|
go install golang.org/x/tools/cmd/goimports
|
||||||
|
|
||||||
# Build from local source.
|
# To pin pluginator, use this recipe instead:
|
||||||
$(MYGOBIN)/gorepomod:
|
# cd api;
|
||||||
cd cmd/gorepomod; \
|
# go install sigs.k8s.io/kustomize/pluginator/v2
|
||||||
go install .
|
|
||||||
|
|
||||||
# Build from local source.
|
|
||||||
$(MYGOBIN)/k8scopy:
|
|
||||||
cd cmd/k8scopy; \
|
|
||||||
go install .
|
|
||||||
|
|
||||||
# Build from local source.
|
|
||||||
$(MYGOBIN)/pluginator:
|
$(MYGOBIN)/pluginator:
|
||||||
cd cmd/pluginator; \
|
cd pluginator; \
|
||||||
go install .
|
go install .
|
||||||
|
|
||||||
# Build from local source.
|
# Install kustomize from whatever is checked out.
|
||||||
$(MYGOBIN)/prchecker:
|
|
||||||
cd cmd/prchecker; \
|
|
||||||
go install .
|
|
||||||
|
|
||||||
# Build from local source.
|
|
||||||
$(MYGOBIN)/kustomize:
|
$(MYGOBIN)/kustomize:
|
||||||
cd kustomize; \
|
cd kustomize; \
|
||||||
go install .
|
go install .
|
||||||
@@ -103,12 +64,8 @@ $(MYGOBIN)/kustomize:
|
|||||||
install-tools: \
|
install-tools: \
|
||||||
$(MYGOBIN)/goimports \
|
$(MYGOBIN)/goimports \
|
||||||
$(MYGOBIN)/golangci-lint-kustomize \
|
$(MYGOBIN)/golangci-lint-kustomize \
|
||||||
$(MYGOBIN)/gorepomod \
|
|
||||||
$(MYGOBIN)/helm \
|
|
||||||
$(MYGOBIN)/k8scopy \
|
|
||||||
$(MYGOBIN)/mdrip \
|
$(MYGOBIN)/mdrip \
|
||||||
$(MYGOBIN)/pluginator \
|
$(MYGOBIN)/pluginator \
|
||||||
$(MYGOBIN)/prchecker \
|
|
||||||
$(MYGOBIN)/stringer
|
$(MYGOBIN)/stringer
|
||||||
|
|
||||||
### Begin kustomize plugin rules.
|
### Begin kustomize plugin rules.
|
||||||
@@ -142,6 +99,7 @@ _builtinplugins = \
|
|||||||
ConfigMapGenerator.go \
|
ConfigMapGenerator.go \
|
||||||
HashTransformer.go \
|
HashTransformer.go \
|
||||||
ImageTagTransformer.go \
|
ImageTagTransformer.go \
|
||||||
|
InventoryTransformer.go \
|
||||||
LabelTransformer.go \
|
LabelTransformer.go \
|
||||||
LegacyOrderTransformer.go \
|
LegacyOrderTransformer.go \
|
||||||
NamespaceTransformer.go \
|
NamespaceTransformer.go \
|
||||||
@@ -150,9 +108,7 @@ _builtinplugins = \
|
|||||||
PatchTransformer.go \
|
PatchTransformer.go \
|
||||||
PrefixSuffixTransformer.go \
|
PrefixSuffixTransformer.go \
|
||||||
ReplicaCountTransformer.go \
|
ReplicaCountTransformer.go \
|
||||||
SecretGenerator.go \
|
SecretGenerator.go
|
||||||
ValueAddTransformer.go \
|
|
||||||
HelmChartInflationGenerator.go
|
|
||||||
|
|
||||||
# Maintaining this explicit list of generated files, and
|
# Maintaining this explicit list of generated files, and
|
||||||
# adding it as a dependency to a few targets, to assure
|
# adding it as a dependency to a few targets, to assure
|
||||||
@@ -168,6 +124,7 @@ $(pGen)/AnnotationsTransformer.go: $(pSrc)/annotationstransformer/AnnotationsTra
|
|||||||
$(pGen)/ConfigMapGenerator.go: $(pSrc)/configmapgenerator/ConfigMapGenerator.go
|
$(pGen)/ConfigMapGenerator.go: $(pSrc)/configmapgenerator/ConfigMapGenerator.go
|
||||||
$(pGen)/HashTransformer.go: $(pSrc)/hashtransformer/HashTransformer.go
|
$(pGen)/HashTransformer.go: $(pSrc)/hashtransformer/HashTransformer.go
|
||||||
$(pGen)/ImageTagTransformer.go: $(pSrc)/imagetagtransformer/ImageTagTransformer.go
|
$(pGen)/ImageTagTransformer.go: $(pSrc)/imagetagtransformer/ImageTagTransformer.go
|
||||||
|
$(pGen)/InventoryTransformer.go: $(pSrc)/inventorytransformer/InventoryTransformer.go
|
||||||
$(pGen)/LabelTransformer.go: $(pSrc)/labeltransformer/LabelTransformer.go
|
$(pGen)/LabelTransformer.go: $(pSrc)/labeltransformer/LabelTransformer.go
|
||||||
$(pGen)/LegacyOrderTransformer.go: $(pSrc)/legacyordertransformer/LegacyOrderTransformer.go
|
$(pGen)/LegacyOrderTransformer.go: $(pSrc)/legacyordertransformer/LegacyOrderTransformer.go
|
||||||
$(pGen)/NamespaceTransformer.go: $(pSrc)/namespacetransformer/NamespaceTransformer.go
|
$(pGen)/NamespaceTransformer.go: $(pSrc)/namespacetransformer/NamespaceTransformer.go
|
||||||
@@ -177,8 +134,6 @@ $(pGen)/PatchTransformer.go: $(pSrc)/patchtransformer/PatchTransformer.go
|
|||||||
$(pGen)/PrefixSuffixTransformer.go: $(pSrc)/prefixsuffixtransformer/PrefixSuffixTransformer.go
|
$(pGen)/PrefixSuffixTransformer.go: $(pSrc)/prefixsuffixtransformer/PrefixSuffixTransformer.go
|
||||||
$(pGen)/ReplicaCountTransformer.go: $(pSrc)/replicacounttransformer/ReplicaCountTransformer.go
|
$(pGen)/ReplicaCountTransformer.go: $(pSrc)/replicacounttransformer/ReplicaCountTransformer.go
|
||||||
$(pGen)/SecretGenerator.go: $(pSrc)/secretgenerator/SecretGenerator.go
|
$(pGen)/SecretGenerator.go: $(pSrc)/secretgenerator/SecretGenerator.go
|
||||||
$(pGen)/ValueAddTransformer.go: $(pSrc)/valueaddtransformer/ValueAddTransformer.go
|
|
||||||
$(pGen)/HelmChartInflationGenerator.go: $(pSrc)/helmchartinflationgenerator/HelmChartInflationGenerator.go
|
|
||||||
|
|
||||||
# The (verbose but portable) Makefile way to convert to lowercase.
|
# The (verbose but portable) Makefile way to convert to lowercase.
|
||||||
toLowerCase = $(subst A,a,$(subst B,b,$(subst C,c,$(subst D,d,$(subst E,e,$(subst F,f,$(subst G,g,$(subst H,h,$(subst I,i,$(subst J,j,$(subst K,k,$(subst L,l,$(subst M,m,$(subst N,n,$(subst O,o,$(subst P,p,$(subst Q,q,$(subst R,r,$(subst S,s,$(subst T,t,$(subst U,u,$(subst V,v,$(subst W,w,$(subst X,x,$(subst Y,y,$(subst Z,z,$1))))))))))))))))))))))))))
|
toLowerCase = $(subst A,a,$(subst B,b,$(subst C,c,$(subst D,d,$(subst E,e,$(subst F,f,$(subst G,g,$(subst H,h,$(subst I,i,$(subst J,j,$(subst K,k,$(subst L,l,$(subst M,m,$(subst N,n,$(subst O,o,$(subst P,p,$(subst Q,q,$(subst R,r,$(subst S,s,$(subst T,t,$(subst U,u,$(subst V,v,$(subst W,w,$(subst X,x,$(subst Y,y,$(subst Z,z,$1))))))))))))))))))))))))))
|
||||||
@@ -197,41 +152,28 @@ $(pGen)/%.go: $(MYGOBIN)/pluginator
|
|||||||
.PHONY: generate-kustomize-builtin-plugins
|
.PHONY: generate-kustomize-builtin-plugins
|
||||||
generate-kustomize-builtin-plugins: $(builtinplugins)
|
generate-kustomize-builtin-plugins: $(builtinplugins)
|
||||||
|
|
||||||
.PHONY: build-kustomize-external-go-plugin
|
.PHONY: kustomize-external-go-plugin-build
|
||||||
build-kustomize-external-go-plugin:
|
kustomize-external-go-plugin-build:
|
||||||
./hack/buildExternalGoPlugins.sh ./plugin
|
./hack/buildExternalGoPlugins.sh ./plugin
|
||||||
|
|
||||||
.PHONY: clean-kustomize-external-go-plugin
|
.PHONY: kustomize-external-go-plugin-clean
|
||||||
clean-kustomize-external-go-plugin:
|
kustomize-external-go-plugin-clean:
|
||||||
./hack/buildExternalGoPlugins.sh ./plugin clean
|
./hack/buildExternalGoPlugins.sh ./plugin clean
|
||||||
|
|
||||||
### End kustomize plugin rules.
|
### End kustomize plugin rules.
|
||||||
|
|
||||||
.PHONY: lint-kustomize
|
.PHONY: lint-kustomize
|
||||||
lint-kustomize: install-tools $(builtinplugins)
|
lint-kustomize: install-tools $(builtinplugins)
|
||||||
cd api; $(MYGOBIN)/golangci-lint-kustomize \
|
cd api; \
|
||||||
-c ../.golangci-kustomize.yml \
|
$(MYGOBIN)/golangci-lint-kustomize -c ../.golangci-kustomize.yml run ./...
|
||||||
run ./...
|
cd kustomize; \
|
||||||
cd kustomize; $(MYGOBIN)/golangci-lint-kustomize \
|
$(MYGOBIN)/golangci-lint-kustomize -c ../.golangci-kustomize.yml run ./...
|
||||||
-c ../.golangci-kustomize.yml \
|
cd pluginator; \
|
||||||
run ./...
|
$(MYGOBIN)/golangci-lint-kustomize -c ../.golangci-kustomize.yml run ./...
|
||||||
cd cmd/pluginator; $(MYGOBIN)/golangci-lint-kustomize \
|
|
||||||
-c ../../.golangci-kustomize.yml \
|
|
||||||
run ./...
|
|
||||||
|
|
||||||
# Used to add non-default compilation flags when experimenting with
|
|
||||||
# plugin-to-api compatibility checks.
|
|
||||||
.PHONY: build-kustomize-api
|
|
||||||
build-kustomize-api: $(builtinplugins)
|
|
||||||
cd api; go build ./...
|
|
||||||
|
|
||||||
.PHONY: generate-kustomize-api
|
|
||||||
generate-kustomize-api: $(MYGOBIN)/k8scopy
|
|
||||||
cd api; go generate ./...
|
|
||||||
|
|
||||||
.PHONY: test-unit-kustomize-api
|
.PHONY: test-unit-kustomize-api
|
||||||
test-unit-kustomize-api: build-kustomize-api
|
test-unit-kustomize-api: $(builtinplugins)
|
||||||
cd api; go test ./... -ldflags "-X sigs.k8s.io/kustomize/api/provenance.version=v444.333.222"
|
cd api; go test ./...
|
||||||
|
|
||||||
.PHONY: test-unit-kustomize-plugins
|
.PHONY: test-unit-kustomize-plugins
|
||||||
test-unit-kustomize-plugins:
|
test-unit-kustomize-plugins:
|
||||||
@@ -247,46 +189,21 @@ test-unit-kustomize-all: \
|
|||||||
test-unit-kustomize-cli \
|
test-unit-kustomize-cli \
|
||||||
test-unit-kustomize-plugins
|
test-unit-kustomize-plugins
|
||||||
|
|
||||||
test-unit-cmd-all:
|
|
||||||
./scripts/kyaml-pre-commit.sh
|
|
||||||
|
|
||||||
test-go-mod:
|
|
||||||
./scripts/check-go-mod.sh
|
|
||||||
|
|
||||||
# Environment variables are defined at
|
|
||||||
# https://github.com/kubernetes/test-infra/blob/master/prow/jobs.md#job-environment-variables
|
|
||||||
.PHONY: test-multi-module
|
|
||||||
test-multi-module: $(MYGOBIN)/prchecker
|
|
||||||
( \
|
|
||||||
export MYGOBIN=$(MYGOBIN); \
|
|
||||||
export REPO_OWNER=$(REPO_OWNER); \
|
|
||||||
export REPO_NAME=$(REPO_NAME); \
|
|
||||||
export PULL_NUMBER=$(PULL_NUMBER); \
|
|
||||||
export MODULES=$(MODULES); \
|
|
||||||
./scripts/check-multi-module.sh; \
|
|
||||||
)
|
|
||||||
|
|
||||||
.PHONY:
|
|
||||||
test-examples-e2e-kustomize: $(MYGOBIN)/mdrip $(MYGOBIN)/kind
|
|
||||||
( \
|
|
||||||
set -e; \
|
|
||||||
/bin/rm -f $(MYGOBIN)/kustomize; \
|
|
||||||
echo "Installing kustomize from ."; \
|
|
||||||
cd kustomize; go install .; cd ..; \
|
|
||||||
./hack/testExamplesE2EAgainstKustomize.sh .; \
|
|
||||||
)
|
|
||||||
|
|
||||||
.PHONY:
|
.PHONY:
|
||||||
test-examples-kustomize-against-HEAD: $(MYGOBIN)/kustomize $(MYGOBIN)/mdrip
|
test-examples-kustomize-against-HEAD: $(MYGOBIN)/kustomize $(MYGOBIN)/mdrip
|
||||||
./hack/testExamplesAgainstKustomize.sh HEAD
|
./hack/testExamplesAgainstKustomize.sh HEAD
|
||||||
|
|
||||||
.PHONY:
|
.PHONY:
|
||||||
test-examples-kustomize-against-3.9: $(MYGOBIN)/mdrip
|
test-examples-kustomize-against-latest: $(MYGOBIN)/mdrip
|
||||||
./hack/testExamplesAgainstKustomize.sh v3.9.3
|
( \
|
||||||
|
set -e; \
|
||||||
.PHONY:
|
/bin/rm -f $(MYGOBIN)/kustomize; \
|
||||||
test-examples-kustomize-against-3.8: $(MYGOBIN)/mdrip
|
echo "Installing kustomize from latest."; \
|
||||||
./hack/testExamplesAgainstKustomize.sh v3.8.10
|
GO111MODULE=on go install sigs.k8s.io/kustomize/kustomize/v3; \
|
||||||
|
./hack/testExamplesAgainstKustomize.sh latest; \
|
||||||
|
echo "Reinstalling kustomize from HEAD."; \
|
||||||
|
cd kustomize; go install .; \
|
||||||
|
)
|
||||||
|
|
||||||
# linux only.
|
# linux only.
|
||||||
# This is for testing an example plugin that
|
# This is for testing an example plugin that
|
||||||
@@ -305,71 +222,29 @@ $(MYGOBIN)/kubeval:
|
|||||||
)
|
)
|
||||||
|
|
||||||
# linux only.
|
# linux only.
|
||||||
# This is for testing an example plugin that uses helm to inflate a chart
|
# This is for testing an example plugin that
|
||||||
# for subsequent kustomization.
|
# uses helm to inflate a chart for subsequent kustomization.
|
||||||
# Don't want to add a hard dependence in go.mod file to helm.
|
# Don't want to add a hard dependence in go.mod file
|
||||||
# Instead, download the binaries.
|
# to helm.
|
||||||
$(MYGOBIN)/helmV2:
|
# Instead, download the binary.
|
||||||
|
$(MYGOBIN)/helm:
|
||||||
( \
|
( \
|
||||||
set -e; \
|
set -e; \
|
||||||
d=$(shell mktemp -d); cd $$d; \
|
d=$(shell mktemp -d); cd $$d; \
|
||||||
tgzFile=helm-v2.13.1-linux-amd64.tar.gz; \
|
wget https://storage.googleapis.com/kubernetes-helm/helm-v2.13.1-linux-amd64.tar.gz; \
|
||||||
wget https://storage.googleapis.com/kubernetes-helm/$$tgzFile; \
|
tar -xvzf helm-v2.13.1-linux-amd64.tar.gz; \
|
||||||
tar -xvzf $$tgzFile; \
|
mv linux-amd64/helm $(MYGOBIN); \
|
||||||
mv linux-amd64/helm $(MYGOBIN)/helmV2; \
|
|
||||||
rm -rf $$d \
|
|
||||||
)
|
|
||||||
|
|
||||||
# Helm V3 differs from helm V2; downloading it to provide coverage for the
|
|
||||||
# chart inflator plugin under helm v3.
|
|
||||||
$(MYGOBIN)/helmV3:
|
|
||||||
( \
|
|
||||||
set -e; \
|
|
||||||
d=$(shell mktemp -d); cd $$d; \
|
|
||||||
tgzFile=helm-v3.4.0-linux-amd64.tar.gz; \
|
|
||||||
wget https://get.helm.sh/$$tgzFile; \
|
|
||||||
tar -xvzf $$tgzFile; \
|
|
||||||
mv linux-amd64/helm $(MYGOBIN)/helmV3; \
|
|
||||||
rm -rf $$d \
|
|
||||||
)
|
|
||||||
|
|
||||||
# Default version of helm is v3.
|
|
||||||
$(MYGOBIN)/helm: $(MYGOBIN)/helmV3
|
|
||||||
ln -s $(MYGOBIN)/helmV3 $(MYGOBIN)/helm
|
|
||||||
|
|
||||||
$(MYGOBIN)/kind:
|
|
||||||
( \
|
|
||||||
set -e; \
|
|
||||||
d=$(shell mktemp -d); cd $$d; \
|
|
||||||
wget -O ./kind https://github.com/kubernetes-sigs/kind/releases/download/v0.7.0/kind-$(shell uname)-amd64; \
|
|
||||||
chmod +x ./kind; \
|
|
||||||
mv ./kind $(MYGOBIN); \
|
|
||||||
rm -rf $$d; \
|
|
||||||
)
|
|
||||||
|
|
||||||
# linux only.
|
|
||||||
$(MYGOBIN)/gh:
|
|
||||||
( \
|
|
||||||
set -e; \
|
|
||||||
d=$(shell mktemp -d); cd $$d; \
|
|
||||||
tgzFile=gh_1.0.0_linux_amd64.tar.gz; \
|
|
||||||
wget https://github.com/cli/cli/releases/download/v1.0.0/$$tgzFile; \
|
|
||||||
tar -xvzf $$tgzFile; \
|
|
||||||
mv gh_1.0.0_linux_amd64/bin/gh $(MYGOBIN)/gh; \
|
|
||||||
rm -rf $$d \
|
rm -rf $$d \
|
||||||
)
|
)
|
||||||
|
|
||||||
.PHONY: clean
|
.PHONY: clean
|
||||||
clean: clean-kustomize-external-go-plugin
|
clean: kustomize-external-go-plugin-clean
|
||||||
go clean --cache
|
go clean --cache
|
||||||
rm -f $(builtinplugins)
|
rm -f $(builtinplugins)
|
||||||
|
rm -f $(MYGOBIN)/pluginator
|
||||||
rm -f $(MYGOBIN)/kustomize
|
rm -f $(MYGOBIN)/kustomize
|
||||||
rm -f $(MYGOBIN)/golangci-lint-kustomize
|
rm -f $(MYGOBIN)/golangci-lint-kustomize
|
||||||
|
|
||||||
# Handle pluginator manually.
|
|
||||||
# rm -f $(MYGOBIN)/pluginator
|
|
||||||
|
|
||||||
# Nuke the site from orbit. It's the only way to be sure.
|
|
||||||
.PHONY: nuke
|
.PHONY: nuke
|
||||||
nuke: clean
|
nuke: clean
|
||||||
go clean --modcache
|
sudo rm -rf $(shell go env GOPATH)/pkg/mod/sigs.k8s.io
|
||||||
|
|||||||
@@ -9,6 +9,3 @@ aliases:
|
|||||||
- mengqiy
|
- mengqiy
|
||||||
- monopole
|
- monopole
|
||||||
- pwittrock
|
- pwittrock
|
||||||
- mortent
|
|
||||||
- phanimarupaka
|
|
||||||
- Shell32-Natsu
|
|
||||||
|
|||||||
90
README.md
90
README.md
@@ -9,31 +9,30 @@ patch [kubernetes style] API objects. It's like
|
|||||||
[`make`], in that what it does is declared in a file,
|
[`make`], in that what it does is declared in a file,
|
||||||
and it's like [`sed`], in that it emits edited text.
|
and it's like [`sed`], in that it emits edited text.
|
||||||
|
|
||||||
This tool is sponsored by [sig-cli] ([KEP]).
|
This tool is sponsored by [sig-cli] ([KEP]), and
|
||||||
|
inspired by [DAM].
|
||||||
|
|
||||||
- [Installation instructions](https://kubernetes-sigs.github.io/kustomize/installation)
|
|
||||||
- [General documentation](https://kubernetes-sigs.github.io/kustomize)
|
|
||||||
- [Examples](examples)
|
|
||||||
|
|
||||||
[](https://prow.k8s.io/job-history/kubernetes-jenkins/pr-logs/directory/kustomize-presubmit-master)
|
[](https://travis-ci.org/kubernetes-sigs/kustomize)
|
||||||
[](https://goreportcard.com/report/github.com/kubernetes-sigs/kustomize)
|
[](https://goreportcard.com/report/github.com/kubernetes-sigs/kustomize)
|
||||||
|
|
||||||
|
Download a binary from the [release page], or see
|
||||||
|
these [instructions](docs/INSTALL.md).
|
||||||
|
|
||||||
|
Browse the [docs](docs) or jump right into the
|
||||||
|
tested [examples](examples).
|
||||||
|
|
||||||
## kubectl integration
|
## kubectl integration
|
||||||
|
|
||||||
The kustomize build flow at [v2.0.3] was added
|
Since [v1.14][kubectl announcement] the kustomize build system has been included in kubectl.
|
||||||
to [kubectl v1.14][kubectl announcement]. The kustomize
|
|
||||||
flow in kubectl has remained frozen at v2.0.3 while work
|
|
||||||
to extract kubectl from the k/k repo, and work to remove
|
|
||||||
kustomize's dependence on core k/k code ([#2506]) has proceeded.
|
|
||||||
The reintegration effort is tracked in [#1500] (and its blocking
|
|
||||||
issues).
|
|
||||||
|
|
||||||
[v2.0.3]: /../../tree/v2.0.3
|
| kubectl version | kustomize version |
|
||||||
[#2506]: https://github.com/kubernetes-sigs/kustomize/issues/2506
|
|---------|--------|
|
||||||
[#1500]: https://github.com/kubernetes-sigs/kustomize/issues/1500
|
| v1.16.0 | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
| v1.15.x | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
| v1.14.x | [v2.0.3](/../../tree/v2.0.3) |
|
||||||
|
|
||||||
For examples and guides for using the kubectl integration please
|
For examples and guides for using the kubectl integration please see the [kubectl book] or the [kubernetes documentation].
|
||||||
see the [kubectl book] or the [kubernetes documentation].
|
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@@ -108,7 +107,7 @@ Take the work from step (1) above, move it into a
|
|||||||
`someApp` subdirectory called `base`, then
|
`someApp` subdirectory called `base`, then
|
||||||
place overlays in a sibling directory.
|
place overlays in a sibling directory.
|
||||||
|
|
||||||
An overlay is just another kustomization, referring to
|
An overlay is just another kustomization, refering to
|
||||||
the base, and referring to patches to apply to that
|
the base, and referring to patches to apply to that
|
||||||
base.
|
base.
|
||||||
|
|
||||||
@@ -134,8 +133,20 @@ The YAML can be directly [applied] to a cluster:
|
|||||||
|
|
||||||
## Community
|
## Community
|
||||||
|
|
||||||
- [file a bug](https://kubernetes-sigs.github.io/kustomize/contributing/bugs/) instructions
|
To file bugs please read [this](docs/bugs.md).
|
||||||
- [contribute a feature](https://kubernetes-sigs.github.io/kustomize/contributing/features/) instructions
|
|
||||||
|
Before working on an implementation, please
|
||||||
|
|
||||||
|
* Read the [eschewed feature list].
|
||||||
|
* File an issue describing
|
||||||
|
how the new feature would behave
|
||||||
|
and label it [kind/feature].
|
||||||
|
|
||||||
|
### Other communication channels
|
||||||
|
|
||||||
|
- [Slack]
|
||||||
|
- [Mailing List]
|
||||||
|
- General kubernetes [community page]
|
||||||
|
|
||||||
### Code of conduct
|
### Code of conduct
|
||||||
|
|
||||||
@@ -144,27 +155,32 @@ is governed by the [Kubernetes Code of Conduct].
|
|||||||
|
|
||||||
[`make`]: https://www.gnu.org/software/make
|
[`make`]: https://www.gnu.org/software/make
|
||||||
[`sed`]: https://www.gnu.org/software/sed
|
[`sed`]: https://www.gnu.org/software/sed
|
||||||
[DAM]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#declarative-application-management
|
[DAM]: docs/glossary.md#declarative-application-management
|
||||||
[KEP]: https://github.com/kubernetes/enhancements/blob/master/keps/sig-cli/0008-kustomize.md
|
[KEP]: https://github.com/kubernetes/enhancements/blob/master/keps/sig-cli/0008-kustomize.md
|
||||||
[Kubernetes Code of Conduct]: code-of-conduct.md
|
[Kubernetes Code of Conduct]: code-of-conduct.md
|
||||||
[applied]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#apply
|
[Mailing List]: https://groups.google.com/forum/#!forum/kubernetes-sig-cli
|
||||||
[base]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#base
|
[Slack]: https://kubernetes.slack.com/messages/sig-cli
|
||||||
[declarative configuration]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#declarative-application-management
|
[applied]: docs/glossary.md#apply
|
||||||
|
[base]: docs/glossary.md#base
|
||||||
|
[community page]: http://kubernetes.io/community/
|
||||||
|
[declarative configuration]: docs/glossary.md#declarative-application-management
|
||||||
|
[eschewed feature list]: docs/eschewedFeatures.md
|
||||||
[imageBase]: docs/images/base.jpg
|
[imageBase]: docs/images/base.jpg
|
||||||
[imageOverlay]: docs/images/overlay.jpg
|
[imageOverlay]: docs/images/overlay.jpg
|
||||||
|
[kind/feature]: /../../labels/kind%2Ffeature
|
||||||
[kubectl announcement]: https://kubernetes.io/blog/2019/03/25/kubernetes-1-14-release-announcement
|
[kubectl announcement]: https://kubernetes.io/blog/2019/03/25/kubernetes-1-14-release-announcement
|
||||||
[kubectl book]: https://kubectl.docs.kubernetes.io/guides/introduction/kustomize/
|
[kubectl book]: https://kubectl.docs.kubernetes.io/pages/app_customization/introduction.html
|
||||||
[kubernetes documentation]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/kustomization/
|
[kubernetes documentation]: https://kubernetes.io/docs/tasks/manage-kubernetes-objects/kustomization/
|
||||||
[kubernetes style]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kubernetes-style-object
|
[kubernetes style]: docs/glossary.md#kubernetes-style-object
|
||||||
[kustomization]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#kustomization
|
[kustomization]: docs/glossary.md#kustomization
|
||||||
[overlay]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#overlay
|
[overlay]: docs/glossary.md#overlay
|
||||||
[overlays]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#overlay
|
[overlays]: docs/glossary.md#overlay
|
||||||
[release page]: https://github.com/kubernetes-sigs/kustomize/releases
|
[release page]: /../../releases
|
||||||
[resource]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#resource
|
[resource]: docs/glossary.md#resource
|
||||||
[resources]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#resource
|
[resources]: docs/glossary.md#resource
|
||||||
[sig-cli]: https://github.com/kubernetes/community/blob/master/sig-cli/README.md
|
[sig-cli]: https://github.com/kubernetes/community/blob/master/sig-cli/README.md
|
||||||
[variant]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#variant
|
[variant]: docs/glossary.md#variant
|
||||||
[variants]: https://kubernetes-sigs.github.io/kustomize/api-reference/glossary#variant
|
[variants]: docs/glossary.md#variant
|
||||||
[v2.0.3]: https://github.com/kubernetes-sigs/kustomize/releases/tag/v2.0.3
|
[v2.0.3]: /../../releases/tag/v2.0.3
|
||||||
[v2.1.0]: https://github.com/kubernetes-sigs/kustomize/releases/tag/v2.1.0
|
[v2.1.0]: /../../releases/tag/v2.1.0
|
||||||
[workflows]: https://kubernetes-sigs.github.io/kustomize/guides
|
[workflows]: docs/workflows.md
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
package builtins
|
package builtins
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sigs.k8s.io/kustomize/api/filters/annotations"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
@@ -24,19 +24,14 @@ func (p *AnnotationsTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *AnnotationsTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *AnnotationsTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if len(p.Annotations) == 0 {
|
t, err := transform.NewMapTransformer(
|
||||||
return nil
|
p.FieldSpecs,
|
||||||
|
p.Annotations,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
for _, r := range m.Resources() {
|
return t.Transform(m)
|
||||||
err := r.ApplyFilter(annotations.Filter{
|
|
||||||
Annotations: p.Annotations,
|
|
||||||
FsSlice: p.FieldSpecs,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAnnotationsTransformerPlugin() resmap.TransformerPlugin {
|
func NewAnnotationsTransformerPlugin() resmap.TransformerPlugin {
|
||||||
|
|||||||
@@ -13,10 +13,13 @@ import (
|
|||||||
type ConfigMapGeneratorPlugin struct {
|
type ConfigMapGeneratorPlugin struct {
|
||||||
h *resmap.PluginHelpers
|
h *resmap.PluginHelpers
|
||||||
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
||||||
|
types.GeneratorOptions
|
||||||
types.ConfigMapArgs
|
types.ConfigMapArgs
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ConfigMapGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte) (err error) {
|
func (p *ConfigMapGeneratorPlugin) Config(
|
||||||
|
h *resmap.PluginHelpers, config []byte) (err error) {
|
||||||
|
p.GeneratorOptions = types.GeneratorOptions{}
|
||||||
p.ConfigMapArgs = types.ConfigMapArgs{}
|
p.ConfigMapArgs = types.ConfigMapArgs{}
|
||||||
err = yaml.Unmarshal(config, p)
|
err = yaml.Unmarshal(config, p)
|
||||||
if p.ConfigMapArgs.Name == "" {
|
if p.ConfigMapArgs.Name == "" {
|
||||||
@@ -31,7 +34,8 @@ func (p *ConfigMapGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte
|
|||||||
|
|
||||||
func (p *ConfigMapGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
func (p *ConfigMapGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
||||||
return p.h.ResmapFactory().FromConfigMapArgs(
|
return p.h.ResmapFactory().FromConfigMapArgs(
|
||||||
kv.NewLoader(p.h.Loader(), p.h.Validator()), p.ConfigMapArgs)
|
kv.NewLoader(p.h.Loader(), p.h.Validator()),
|
||||||
|
&p.GeneratorOptions, p.ConfigMapArgs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewConfigMapGeneratorPlugin() resmap.GeneratorPlugin {
|
func NewConfigMapGeneratorPlugin() resmap.GeneratorPlugin {
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ func (p *HashTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
res.StorePreviousId()
|
|
||||||
res.SetName(fmt.Sprintf("%s-%s", res.GetName(), h))
|
res.SetName(fmt.Sprintf("%s-%s", res.GetName(), h))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,287 +0,0 @@
|
|||||||
// Code generated by pluginator on HelmChartInflationGenerator; DO NOT EDIT.
|
|
||||||
// pluginator {unknown 1970-01-01T00:00:00Z }
|
|
||||||
|
|
||||||
package builtins
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
|
||||||
"regexp"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/imdario/mergo"
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// HelmChartInflationGeneratorPlugin is a plugin to generate resources
|
|
||||||
// from a remote or local helm chart.
|
|
||||||
type HelmChartInflationGeneratorPlugin struct {
|
|
||||||
h *resmap.PluginHelpers
|
|
||||||
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
|
||||||
runHelmCommand func([]string) ([]byte, error)
|
|
||||||
types.HelmChartArgs
|
|
||||||
tmpDir string
|
|
||||||
}
|
|
||||||
|
|
||||||
var KustomizePlugin HelmChartInflationGeneratorPlugin
|
|
||||||
|
|
||||||
// Config uses the input plugin configurations `config` to setup the generator
|
|
||||||
// options
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte) error {
|
|
||||||
p.h = h
|
|
||||||
err := yaml.Unmarshal(config, p)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
tmpDir, err := filesys.NewTmpConfirmedDir()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.tmpDir = string(tmpDir)
|
|
||||||
if p.ChartName == "" {
|
|
||||||
return fmt.Errorf("chartName cannot be empty")
|
|
||||||
}
|
|
||||||
if p.ChartHome == "" {
|
|
||||||
p.ChartHome = filepath.Join(p.tmpDir, "chart")
|
|
||||||
}
|
|
||||||
if p.ChartRepoName == "" {
|
|
||||||
p.ChartRepoName = "stable"
|
|
||||||
}
|
|
||||||
if p.HelmBin == "" {
|
|
||||||
p.HelmBin = "helm"
|
|
||||||
}
|
|
||||||
if p.HelmHome == "" {
|
|
||||||
p.HelmHome = filepath.Join(p.tmpDir, ".helm")
|
|
||||||
}
|
|
||||||
if p.Values == "" {
|
|
||||||
p.Values = filepath.Join(p.ChartHome, p.ChartName, "values.yaml")
|
|
||||||
}
|
|
||||||
if p.ValuesMerge == "" {
|
|
||||||
p.ValuesMerge = "override"
|
|
||||||
}
|
|
||||||
// runHelmCommand will run `helm` command with args provided. Return stdout
|
|
||||||
// and error if there is any.
|
|
||||||
p.runHelmCommand = func(args []string) ([]byte, error) {
|
|
||||||
stdout := new(bytes.Buffer)
|
|
||||||
stderr := new(bytes.Buffer)
|
|
||||||
cmd := exec.Command(p.HelmBin, args...)
|
|
||||||
cmd.Stdout = stdout
|
|
||||||
cmd.Stderr = stderr
|
|
||||||
cmd.Env = append(cmd.Env,
|
|
||||||
fmt.Sprintf("HELM_CONFIG_HOME=%s", p.HelmHome),
|
|
||||||
fmt.Sprintf("HELM_CACHE_HOME=%s/.cache", p.HelmHome),
|
|
||||||
fmt.Sprintf("HELM_DATA_HOME=%s/.data", p.HelmHome),
|
|
||||||
)
|
|
||||||
err := cmd.Run()
|
|
||||||
if err != nil {
|
|
||||||
return stdout.Bytes(),
|
|
||||||
errors.Wrap(
|
|
||||||
fmt.Errorf("failed to run command %s %s", p.HelmBin, strings.Join(args, " ")),
|
|
||||||
stderr.String(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
return stdout.Bytes(), nil
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// EncodeValues for writing
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) EncodeValues(w io.Writer) error {
|
|
||||||
d, err := yaml.Marshal(p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, err = w.Write(d)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// useValuesLocal process (merge) inflator config provided values with chart default values.yaml
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) useValuesLocal() error {
|
|
||||||
// not override, merge, none
|
|
||||||
if !(p.ValuesMerge == "none" || p.ValuesMerge == "no" || p.ValuesMerge == "false") {
|
|
||||||
var pValues []byte
|
|
||||||
var err error
|
|
||||||
|
|
||||||
if filepath.IsAbs(p.Values) {
|
|
||||||
pValues, err = ioutil.ReadFile(p.Values)
|
|
||||||
} else {
|
|
||||||
pValues, err = p.h.Loader().Load(p.Values)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
chValues := make(map[string]interface{})
|
|
||||||
err = yaml.Unmarshal(pValues, &chValues)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if p.ValuesMerge == "override" {
|
|
||||||
err = mergo.Merge(&chValues, p.ValuesLocal, mergo.WithOverride)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if p.ValuesMerge == "merge" {
|
|
||||||
err = mergo.Merge(&chValues, p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
p.ValuesLocal = chValues
|
|
||||||
}
|
|
||||||
b, err := yaml.Marshal(p.ValuesLocal)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
path, err := p.writeValuesBytes(b)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Values = path
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// copyValues will copy the relative values file into the temp directory
|
|
||||||
// to avoid messing up with CWD.
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) copyValues() error {
|
|
||||||
// only copy when the values path is not absolute
|
|
||||||
if filepath.IsAbs(p.Values) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
// we must use use loader to read values file
|
|
||||||
b, err := p.h.Loader().Load(p.Values)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
path, err := p.writeValuesBytes(b)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Values = path
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) writeValuesBytes(b []byte) (string, error) {
|
|
||||||
path := filepath.Join(p.ChartHome, p.ChartName, "kustomize-values.yaml")
|
|
||||||
err := ioutil.WriteFile(path, b, 0644)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return path, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate implements generator
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
|
||||||
// cleanup
|
|
||||||
defer os.RemoveAll(p.tmpDir)
|
|
||||||
// check helm version. we only support V3
|
|
||||||
err := p.checkHelmVersion()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
// pull the chart
|
|
||||||
if !p.checkLocalChart() {
|
|
||||||
_, err := p.runHelmCommand(p.getPullCommandArgs())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// inflator config valuesLocal
|
|
||||||
if len(p.ValuesLocal) > 0 {
|
|
||||||
err := p.useValuesLocal()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
err := p.copyValues()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// render the charts
|
|
||||||
stdout, err := p.runHelmCommand(p.getTemplateCommandArgs())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return p.h.ResmapFactory().NewResMapFromBytes(stdout)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) getTemplateCommandArgs() []string {
|
|
||||||
args := []string{"template"}
|
|
||||||
if p.ReleaseName != "" {
|
|
||||||
args = append(args, p.ReleaseName)
|
|
||||||
}
|
|
||||||
args = append(args, filepath.Join(p.ChartHome, p.ChartName))
|
|
||||||
if p.ReleaseNamespace != "" {
|
|
||||||
args = append(args, "--namespace", p.ReleaseNamespace)
|
|
||||||
}
|
|
||||||
if p.Values != "" {
|
|
||||||
args = append(args, "--values", p.Values)
|
|
||||||
}
|
|
||||||
args = append(args, p.ExtraArgs...)
|
|
||||||
return args
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) getPullCommandArgs() []string {
|
|
||||||
args := []string{"pull", "--untar", "--untardir", p.ChartHome}
|
|
||||||
chartName := fmt.Sprintf("%s/%s", p.ChartRepoName, p.ChartName)
|
|
||||||
if p.ChartVersion != "" {
|
|
||||||
args = append(args, "--version", p.ChartVersion)
|
|
||||||
}
|
|
||||||
if p.ChartRepoURL != "" {
|
|
||||||
args = append(args, "--repo", p.ChartRepoURL)
|
|
||||||
chartName = p.ChartName
|
|
||||||
}
|
|
||||||
|
|
||||||
args = append(args, chartName)
|
|
||||||
|
|
||||||
return args
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkLocalChart will return true if the chart does exist in
|
|
||||||
// local chart home.
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) checkLocalChart() bool {
|
|
||||||
path := filepath.Join(p.ChartHome, p.ChartName)
|
|
||||||
s, err := os.Stat(path)
|
|
||||||
if err != nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return s.IsDir()
|
|
||||||
}
|
|
||||||
|
|
||||||
// checkHelmVersion will return an error if the helm version is not V3
|
|
||||||
func (p *HelmChartInflationGeneratorPlugin) checkHelmVersion() error {
|
|
||||||
stdout, err := p.runHelmCommand([]string{"version", "-c", "--short"})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
r, err := regexp.Compile(`v\d+(\.\d+)+`)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
v := string(r.Find(stdout))[1:]
|
|
||||||
majorVersion := strings.Split(v, ".")[0]
|
|
||||||
if majorVersion != "3" {
|
|
||||||
return fmt.Errorf("this plugin requires helm V3 but got v%s", v)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewHelmChartInflationGeneratorPlugin() resmap.GeneratorPlugin {
|
|
||||||
return &HelmChartInflationGeneratorPlugin{}
|
|
||||||
}
|
|
||||||
@@ -8,9 +8,10 @@ import (
|
|||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/imagetag"
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -30,19 +31,18 @@ func (p *ImageTagTransformerPlugin) Config(
|
|||||||
|
|
||||||
func (p *ImageTagTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *ImageTagTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
// traverse all fields at first
|
for _, path := range p.FieldSpecs {
|
||||||
err := r.ApplyFilter(imagetag.LegacyFilter{
|
if !r.OrgId().IsSelected(&path.Gvk) {
|
||||||
ImageTag: p.ImageTag,
|
continue
|
||||||
})
|
}
|
||||||
if err != nil {
|
err := transform.MutateField(
|
||||||
return err
|
r.Map(), path.PathSlice(), false, p.mutateImage)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// then use user specified field specs
|
// Kept for backward compatibility
|
||||||
err = r.ApplyFilter(imagetag.Filter{
|
if err := p.findAndReplaceImage(r.Map()); err != nil && r.OrgId().Kind != `CustomResourceDefinition` {
|
||||||
ImageTag: p.ImageTag,
|
|
||||||
FsSlice: p.FieldSpecs,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -144,10 +144,8 @@ func (p *ImageTagTransformerPlugin) findContainers(obj map[string]interface{}) e
|
|||||||
}
|
}
|
||||||
|
|
||||||
func isImageMatched(s, t string) bool {
|
func isImageMatched(s, t string) bool {
|
||||||
// Tag values are limited to [a-zA-Z0-9_.{}-].
|
// Tag values are limited to [a-zA-Z0-9_.-].
|
||||||
// Some tools like Bazel rules_k8s allow tag patterns with {} characters.
|
pattern, _ := regexp.Compile("^" + t + "(@sha256)?(:[a-zA-Z0-9_.-]*)?$")
|
||||||
// More info: https://github.com/bazelbuild/rules_k8s/pull/423
|
|
||||||
pattern, _ := regexp.Compile("^" + t + "(@sha256)?(:[a-zA-Z0-9_.{}-]*)?$")
|
|
||||||
return pattern.MatchString(s)
|
return pattern.MatchString(s)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
128
api/builtins/InventoryTransformer.go
Normal file
128
api/builtins/InventoryTransformer.go
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
// Code generated by pluginator on InventoryTransformer; DO NOT EDIT.
|
||||||
|
// pluginator {unknown 1970-01-01T00:00:00Z }
|
||||||
|
|
||||||
|
package builtins
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/hasher"
|
||||||
|
"sigs.k8s.io/kustomize/api/inventory"
|
||||||
|
"sigs.k8s.io/kustomize/api/kv"
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
"sigs.k8s.io/yaml"
|
||||||
|
)
|
||||||
|
|
||||||
|
type InventoryTransformerPlugin struct {
|
||||||
|
h *resmap.PluginHelpers
|
||||||
|
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
||||||
|
Policy string `json:"policy,omitempty" yaml:"policy,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *InventoryTransformerPlugin) Config(
|
||||||
|
h *resmap.PluginHelpers, c []byte) (err error) {
|
||||||
|
p.h = h
|
||||||
|
err = yaml.Unmarshal(c, p)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if p.Policy == "" {
|
||||||
|
p.Policy = types.GarbageIgnore.String()
|
||||||
|
}
|
||||||
|
if p.Policy != types.GarbageCollect.String() &&
|
||||||
|
p.Policy != types.GarbageIgnore.String() {
|
||||||
|
return fmt.Errorf(
|
||||||
|
"unrecognized garbagePolicy '%s'", p.Policy)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transform generates an inventory object from the input ResMap.
|
||||||
|
// This ConfigMap supports the pruning command in
|
||||||
|
// the client side tool proposed here:
|
||||||
|
// https://github.com/kubernetes/enhancements/pull/810
|
||||||
|
//
|
||||||
|
// The inventory data is written to the ConfigMap's
|
||||||
|
// annotations, rather than to the key-value pairs in
|
||||||
|
// the ConfigMap's data field, since
|
||||||
|
// 1. Keys in a ConfigMap's data field are too
|
||||||
|
// constrained for this purpose.
|
||||||
|
// 2. Using annotations allow any object to be used,
|
||||||
|
// not just a ConfigMap, should some other object
|
||||||
|
// (e.g. some App object) become more desirable
|
||||||
|
// for this purpose.
|
||||||
|
func (p *InventoryTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
|
|
||||||
|
inv, h, err := makeInventory(m)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
args := types.ConfigMapArgs{}
|
||||||
|
args.Name = p.Name
|
||||||
|
args.Namespace = p.Namespace
|
||||||
|
opts := &types.GeneratorOptions{
|
||||||
|
Annotations: make(map[string]string),
|
||||||
|
}
|
||||||
|
opts.Annotations[inventory.HashAnnotation] = h
|
||||||
|
err = inv.UpdateAnnotations(opts.Annotations)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
cm, err := p.h.ResmapFactory().RF().MakeConfigMap(
|
||||||
|
kv.NewLoader(p.h.Loader(), p.h.Validator()), opts, &args)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if p.Policy == types.GarbageCollect.String() {
|
||||||
|
for _, byeBye := range m.AllIds() {
|
||||||
|
m.Remove(byeBye)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return m.Append(cm)
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeInventory(m resmap.ResMap) (
|
||||||
|
inv *inventory.Inventory, hash string, err error) {
|
||||||
|
inv = inventory.NewInventory()
|
||||||
|
var keys []string
|
||||||
|
for _, r := range m.Resources() {
|
||||||
|
ns := r.GetNamespace()
|
||||||
|
item := resid.NewResIdWithNamespace(r.GetGvk(), r.GetName(), ns)
|
||||||
|
if _, ok := inv.Current[item]; ok {
|
||||||
|
return nil, "", fmt.Errorf(
|
||||||
|
"item '%v' already in inventory", item)
|
||||||
|
}
|
||||||
|
inv.Current[item], err = computeRefs(r, m)
|
||||||
|
if err != nil {
|
||||||
|
return nil, "", err
|
||||||
|
}
|
||||||
|
keys = append(keys, item.String())
|
||||||
|
}
|
||||||
|
h, err := hasher.SortArrayAndComputeHash(keys)
|
||||||
|
return inv, h, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func computeRefs(
|
||||||
|
r *resource.Resource, m resmap.ResMap) (refs []resid.ResId, err error) {
|
||||||
|
for _, refid := range r.GetRefBy() {
|
||||||
|
ref, err := m.GetByCurrentId(refid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
refs = append(
|
||||||
|
refs,
|
||||||
|
resid.NewResIdWithNamespace(
|
||||||
|
ref.GetGvk(), ref.GetName(), ref.GetNamespace()))
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewInventoryTransformerPlugin() resmap.TransformerPlugin {
|
||||||
|
return &InventoryTransformerPlugin{}
|
||||||
|
}
|
||||||
@@ -4,8 +4,8 @@
|
|||||||
package builtins
|
package builtins
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"sigs.k8s.io/kustomize/api/filters/labels"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
@@ -24,19 +24,14 @@ func (p *LabelTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *LabelTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *LabelTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if len(p.Labels) == 0 {
|
t, err := transform.NewMapTransformer(
|
||||||
return nil
|
p.FieldSpecs,
|
||||||
|
p.Labels,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
for _, r := range m.Resources() {
|
return t.Transform(m)
|
||||||
err := r.ApplyFilter(labels.Filter{
|
|
||||||
Labels: p.Labels,
|
|
||||||
FsSlice: p.FieldSpecs,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewLabelTransformerPlugin() resmap.TransformerPlugin {
|
func NewLabelTransformerPlugin() resmap.TransformerPlugin {
|
||||||
|
|||||||
@@ -6,8 +6,11 @@ package builtins
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/namespace"
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
@@ -30,27 +33,99 @@ func (p *NamespaceTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
if r.IsEmpty() {
|
if len(r.Map()) == 0 {
|
||||||
// Don't mutate empty objects?
|
// Don't mutate empty objects?
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
r.StorePreviousId()
|
|
||||||
err := r.ApplyFilter(namespace.Filter{
|
id := r.OrgId()
|
||||||
Namespace: p.Namespace,
|
applicableFs := p.applicableFieldSpecs(id)
|
||||||
FsSlice: p.FieldSpecs,
|
|
||||||
})
|
for _, fs := range applicableFs {
|
||||||
if err != nil {
|
err := transform.MutateField(
|
||||||
return err
|
r.Map(), fs.PathSlice(), fs.CreateIfNotPresent,
|
||||||
|
p.changeNamespace(r))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
matches := m.GetMatchingResourcesByCurrentId(r.CurId().Equals)
|
matches := m.GetMatchingResourcesByCurrentId(r.CurId().Equals)
|
||||||
if len(matches) != 1 {
|
if len(matches) != 1 {
|
||||||
return fmt.Errorf(
|
return fmt.Errorf("namespace tranformation produces ID conflict: %#v", matches)
|
||||||
"namespace transformation produces ID conflict: %+v", matches)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const metaNamespace = "metadata/namespace"
|
||||||
|
|
||||||
|
// Special casing metadata.namespace since
|
||||||
|
// all objects have it, even "ClusterKind" objects
|
||||||
|
// that don't exist in a namespace (the Namespace
|
||||||
|
// object itself doesn't live in a namespace).
|
||||||
|
func (p *NamespaceTransformerPlugin) applicableFieldSpecs(id resid.ResId) []types.FieldSpec {
|
||||||
|
var res []types.FieldSpec
|
||||||
|
for _, fs := range p.FieldSpecs {
|
||||||
|
if id.IsSelected(&fs.Gvk) && (fs.Path != metaNamespace || (fs.Path == metaNamespace && id.IsNamespaceableKind())) {
|
||||||
|
res = append(res, fs)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *NamespaceTransformerPlugin) changeNamespace(
|
||||||
|
_ *resource.Resource) func(in interface{}) (interface{}, error) {
|
||||||
|
return func(in interface{}) (interface{}, error) {
|
||||||
|
switch in.(type) {
|
||||||
|
case string:
|
||||||
|
// will happen when the metadata/namespace
|
||||||
|
// value is replaced
|
||||||
|
return p.Namespace, nil
|
||||||
|
case []interface{}:
|
||||||
|
l, _ := in.([]interface{})
|
||||||
|
for idx, item := range l {
|
||||||
|
switch item.(type) {
|
||||||
|
case map[string]interface{}:
|
||||||
|
// Will happen when mutating the subjects
|
||||||
|
// field of ClusterRoleBinding and RoleBinding
|
||||||
|
inMap, _ := item.(map[string]interface{})
|
||||||
|
if _, ok := inMap["name"]; !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
name, ok := inMap["name"].(string)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// The only case we need to force the namespace
|
||||||
|
// if for the "service account". "default" is
|
||||||
|
// kind of hardcoded here for right now.
|
||||||
|
if name != "default" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
inMap["namespace"] = p.Namespace
|
||||||
|
l[idx] = inMap
|
||||||
|
default:
|
||||||
|
// nothing to do for right now
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return in, nil
|
||||||
|
case map[string]interface{}:
|
||||||
|
// Will happen if the createField=true
|
||||||
|
// when the namespace is added to the
|
||||||
|
// object
|
||||||
|
inMap := in.(map[string]interface{})
|
||||||
|
if len(inMap) == 0 {
|
||||||
|
return p.Namespace, nil
|
||||||
|
} else {
|
||||||
|
return in, nil
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return in, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func NewNamespaceTransformerPlugin() resmap.TransformerPlugin {
|
func NewNamespaceTransformerPlugin() resmap.TransformerPlugin {
|
||||||
return &NamespaceTransformerPlugin{}
|
return &NamespaceTransformerPlugin{}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,8 +8,8 @@ import (
|
|||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
jsonpatch "github.com/evanphx/json-patch"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"sigs.k8s.io/kustomize/api/filters/patchjson6902"
|
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
@@ -18,9 +18,9 @@ import (
|
|||||||
type PatchJson6902TransformerPlugin struct {
|
type PatchJson6902TransformerPlugin struct {
|
||||||
ldr ifc.Loader
|
ldr ifc.Loader
|
||||||
decodedPatch jsonpatch.Patch
|
decodedPatch jsonpatch.Patch
|
||||||
Target *types.Selector `json:"target,omitempty" yaml:"target,omitempty"`
|
Target types.PatchTarget `json:"target,omitempty" yaml:"target,omitempty"`
|
||||||
Path string `json:"path,omitempty" yaml:"path,omitempty"`
|
Path string `json:"path,omitempty" yaml:"path,omitempty"`
|
||||||
JsonOp string `json:"jsonOp,omitempty" yaml:"jsonOp,omitempty"`
|
JsonOp string `json:"jsonOp,omitempty" yaml:"jsonOp,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchJson6902TransformerPlugin) Config(
|
func (p *PatchJson6902TransformerPlugin) Config(
|
||||||
@@ -70,22 +70,29 @@ func (p *PatchJson6902TransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchJson6902TransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *PatchJson6902TransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if p.Target == nil {
|
id := resid.NewResIdWithNamespace(
|
||||||
return fmt.Errorf("must specify a target for patch %s", p.JsonOp)
|
resid.Gvk{
|
||||||
}
|
Group: p.Target.Group,
|
||||||
resources, err := m.Select(*p.Target)
|
Version: p.Target.Version,
|
||||||
|
Kind: p.Target.Kind,
|
||||||
|
},
|
||||||
|
p.Target.Name,
|
||||||
|
p.Target.Namespace,
|
||||||
|
)
|
||||||
|
obj, err := m.GetById(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for _, res := range resources {
|
rawObj, err := obj.MarshalJSON()
|
||||||
err = res.ApplyFilter(patchjson6902.Filter{
|
if err != nil {
|
||||||
Patch: p.JsonOp,
|
return err
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
modifiedObj, err := p.decodedPatch.Apply(rawObj)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(
|
||||||
|
err, "failed to apply json patch '%s'", p.JsonOp)
|
||||||
|
}
|
||||||
|
return obj.UnmarshalJSON(modifiedObj)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewPatchJson6902TransformerPlugin() resmap.TransformerPlugin {
|
func NewPatchJson6902TransformerPlugin() resmap.TransformerPlugin {
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type PatchStrategicMergeTransformerPlugin struct {
|
type PatchStrategicMergeTransformerPlugin struct {
|
||||||
|
h *resmap.PluginHelpers
|
||||||
loadedPatches []*resource.Resource
|
loadedPatches []*resource.Resource
|
||||||
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
Paths []types.PatchStrategicMerge `json:"paths,omitempty" yaml:"paths,omitempty"`
|
||||||
Patches string `json:"patches,omitempty" yaml:"patches,omitempty"`
|
Patches string `json:"patches,omitempty" yaml:"patches,omitempty"`
|
||||||
@@ -20,6 +21,7 @@ type PatchStrategicMergeTransformerPlugin struct {
|
|||||||
|
|
||||||
func (p *PatchStrategicMergeTransformerPlugin) Config(
|
func (p *PatchStrategicMergeTransformerPlugin) Config(
|
||||||
h *resmap.PluginHelpers, c []byte) (err error) {
|
h *resmap.PluginHelpers, c []byte) (err error) {
|
||||||
|
p.h = h
|
||||||
err = yaml.Unmarshal(c, p)
|
err = yaml.Unmarshal(c, p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -29,18 +31,13 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
if len(p.Paths) != 0 {
|
if len(p.Paths) != 0 {
|
||||||
for _, onePath := range p.Paths {
|
for _, onePath := range p.Paths {
|
||||||
// The following oddly attempts to interpret a path string as an
|
res, err := p.h.ResmapFactory().RF().SliceFromBytes([]byte(onePath))
|
||||||
// actual patch (instead of as a path to a file containing a patch).
|
|
||||||
// All tests pass if this code is commented out. This code should
|
|
||||||
// be deleted; the user should use the Patches field which
|
|
||||||
// exists for this purpose (inline patch declaration).
|
|
||||||
res, err := h.ResmapFactory().RF().SliceFromBytes([]byte(onePath))
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
p.loadedPatches = append(p.loadedPatches, res...)
|
p.loadedPatches = append(p.loadedPatches, res...)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
res, err = h.ResmapFactory().RF().SliceFromPatches(
|
res, err = p.h.ResmapFactory().RF().SliceFromPatches(
|
||||||
h.Loader(), []types.PatchStrategicMerge{onePath})
|
p.h.Loader(), []types.PatchStrategicMerge{onePath})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -48,7 +45,7 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if p.Patches != "" {
|
if p.Patches != "" {
|
||||||
res, err := h.ResmapFactory().RF().SliceFromBytes([]byte(p.Patches))
|
res, err := p.h.ResmapFactory().RF().SliceFromBytes([]byte(p.Patches))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -59,24 +56,31 @@ func (p *PatchStrategicMergeTransformerPlugin) Config(
|
|||||||
return fmt.Errorf(
|
return fmt.Errorf(
|
||||||
"patch appears to be empty; files=%v, Patch=%s", p.Paths, p.Patches)
|
"patch appears to be empty; files=%v, Patch=%s", p.Paths, p.Patches)
|
||||||
}
|
}
|
||||||
// Merge the patches, looking for conflicts.
|
return err
|
||||||
_, err = h.ResmapFactory().ConflatePatches(p.loadedPatches)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchStrategicMergeTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *PatchStrategicMergeTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
for _, patch := range p.loadedPatches {
|
patches, err := p.h.ResmapFactory().MergePatches(p.loadedPatches)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, patch := range patches.Resources() {
|
||||||
target, err := m.GetById(patch.OrgId())
|
target, err := m.GetById(patch.OrgId())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if err = m.ApplySmPatch(
|
err = target.Patch(patch.Kunstructured)
|
||||||
resource.MakeIdSet([]*resource.Resource{target}), patch); err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
// remove the resource from resmap
|
||||||
|
// when the patch is to $patch: delete that target
|
||||||
|
if len(target.Map()) == 0 {
|
||||||
|
err = m.Remove(target.CurId())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,10 +5,9 @@ package builtins
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
|
||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
jsonpatch "github.com/evanphx/json-patch"
|
||||||
"sigs.k8s.io/kustomize/api/filters/patchjson6902"
|
"github.com/pkg/errors"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
@@ -24,92 +23,99 @@ type PatchTransformerPlugin struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchTransformerPlugin) Config(
|
func (p *PatchTransformerPlugin) Config(
|
||||||
h *resmap.PluginHelpers, c []byte) error {
|
h *resmap.PluginHelpers, c []byte) (err error) {
|
||||||
err := yaml.Unmarshal(c, p)
|
err = yaml.Unmarshal(c, p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
p.Patch = strings.TrimSpace(p.Patch)
|
|
||||||
if p.Patch == "" && p.Path == "" {
|
if p.Patch == "" && p.Path == "" {
|
||||||
return fmt.Errorf(
|
err = fmt.Errorf(
|
||||||
"must specify one of patch and path in\n%s", string(c))
|
"must specify one of patch and path in\n%s", string(c))
|
||||||
|
return
|
||||||
}
|
}
|
||||||
if p.Patch != "" && p.Path != "" {
|
if p.Patch != "" && p.Path != "" {
|
||||||
return fmt.Errorf(
|
err = fmt.Errorf(
|
||||||
"patch and path can't be set at the same time\n%s", string(c))
|
"patch and path can't be set at the same time\n%s", string(c))
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
var in []byte
|
||||||
if p.Path != "" {
|
if p.Path != "" {
|
||||||
loaded, loadErr := h.Loader().Load(p.Path)
|
in, err = h.Loader().Load(p.Path)
|
||||||
if loadErr != nil {
|
if err != nil {
|
||||||
return loadErr
|
return
|
||||||
}
|
}
|
||||||
p.Patch = string(loaded)
|
}
|
||||||
|
if p.Patch != "" {
|
||||||
|
in = []byte(p.Patch)
|
||||||
}
|
}
|
||||||
|
|
||||||
patchSM, errSM := h.ResmapFactory().RF().FromBytes([]byte(p.Patch))
|
patchSM, errSM := h.ResmapFactory().RF().FromBytes(in)
|
||||||
patchJson, errJson := jsonPatchFromBytes([]byte(p.Patch))
|
patchJson, errJson := jsonPatchFromBytes(in)
|
||||||
if (errSM == nil && errJson == nil) ||
|
|
||||||
(patchSM != nil && patchJson != nil) {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"illegally qualifies as both an SM and JSON patch: [%v]",
|
|
||||||
p.Patch)
|
|
||||||
}
|
|
||||||
if errSM != nil && errJson != nil {
|
if errSM != nil && errJson != nil {
|
||||||
return fmt.Errorf(
|
err = fmt.Errorf(
|
||||||
"unable to parse SM or JSON patch from [%v]", p.Patch)
|
"unable to get either a Strategic Merge Patch or JSON patch 6902 from %s", p.Patch)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
if errSM == nil {
|
if errSM == nil && errJson != nil {
|
||||||
p.loadedPatch = patchSM
|
p.loadedPatch = patchSM
|
||||||
} else {
|
}
|
||||||
|
if errJson == nil && errSM != nil {
|
||||||
p.decodedPatch = patchJson
|
p.decodedPatch = patchJson
|
||||||
}
|
}
|
||||||
|
if patchSM != nil && patchJson != nil {
|
||||||
|
err = fmt.Errorf(
|
||||||
|
"a patch can't be both a Strategic Merge Patch and JSON patch 6902 %s", p.Patch)
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PatchTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *PatchTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
if p.loadedPatch == nil {
|
if p.loadedPatch != nil && p.Target == nil {
|
||||||
return p.transformJson6902(m, p.decodedPatch)
|
target, err := m.GetById(p.loadedPatch.OrgId())
|
||||||
} else {
|
|
||||||
// The patch was a strategic merge patch
|
|
||||||
return p.transformStrategicMerge(m, p.loadedPatch)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// transformStrategicMerge applies the provided strategic merge patch
|
|
||||||
// to all the resources in the ResMap that match either the Target or
|
|
||||||
// the identifier of the patch.
|
|
||||||
func (p *PatchTransformerPlugin) transformStrategicMerge(m resmap.ResMap, patch *resource.Resource) error {
|
|
||||||
if p.Target == nil {
|
|
||||||
target, err := m.GetById(patch.OrgId())
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return target.ApplySmPatch(patch)
|
err = target.Patch(p.loadedPatch.Kunstructured)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
selected, err := m.Select(*p.Target)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return m.ApplySmPatch(resource.MakeIdSet(selected), patch)
|
|
||||||
}
|
|
||||||
|
|
||||||
// transformJson6902 applies the provided json6902 patch
|
|
||||||
// to all the resources in the ResMap that match the Target.
|
|
||||||
func (p *PatchTransformerPlugin) transformJson6902(m resmap.ResMap, patch jsonpatch.Patch) error {
|
|
||||||
if p.Target == nil {
|
if p.Target == nil {
|
||||||
return fmt.Errorf("must specify a target for patch %s", p.Patch)
|
return fmt.Errorf("must specify a target for patch %s", p.Patch)
|
||||||
}
|
}
|
||||||
|
|
||||||
resources, err := m.Select(*p.Target)
|
resources, err := m.Select(*p.Target)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for _, res := range resources {
|
for _, res := range resources {
|
||||||
res.StorePreviousId()
|
if p.decodedPatch != nil {
|
||||||
err = res.ApplyFilter(patchjson6902.Filter{
|
rawObj, err := res.MarshalJSON()
|
||||||
Patch: p.Patch,
|
if err != nil {
|
||||||
})
|
return err
|
||||||
if err != nil {
|
}
|
||||||
return err
|
modifiedObj, err := p.decodedPatch.Apply(rawObj)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrapf(
|
||||||
|
err, "failed to apply json patch '%s'", p.Patch)
|
||||||
|
}
|
||||||
|
err = res.UnmarshalJSON(modifiedObj)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if p.loadedPatch != nil {
|
||||||
|
patchCopy := p.loadedPatch.DeepCopy()
|
||||||
|
patchCopy.SetName(res.GetName())
|
||||||
|
patchCopy.SetNamespace(res.GetNamespace())
|
||||||
|
patchCopy.SetGvk(res.GetGvk())
|
||||||
|
err = res.Patch(patchCopy.Kunstructured)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
|||||||
@@ -5,27 +5,31 @@ package builtins
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/prefixsuffix"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Add the given prefix and suffix to the field.
|
// Add the given prefix and suffix to the field.
|
||||||
type PrefixSuffixTransformerPlugin struct {
|
type PrefixSuffixTransformerPlugin struct {
|
||||||
Prefix string `json:"prefix,omitempty" yaml:"prefix,omitempty"`
|
Prefix string `json:"prefix,omitempty" yaml:"prefix,omitempty"`
|
||||||
Suffix string `json:"suffix,omitempty" yaml:"suffix,omitempty"`
|
Suffix string `json:"suffix,omitempty" yaml:"suffix,omitempty"`
|
||||||
FieldSpecs types.FsSlice `json:"fieldSpecs,omitempty" yaml:"fieldSpecs,omitempty"`
|
FieldSpecs []types.FieldSpec `json:"fieldSpecs,omitempty" yaml:"fieldSpecs,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// A Gvk skip list for prefix/suffix modification.
|
// Not placed in a file yet due to lack of demand.
|
||||||
// hard coded for now - eventually should be part of config.
|
var prefixSuffixFieldSpecsToSkip = []types.FieldSpec{
|
||||||
var prefixSuffixFieldSpecsToSkip = types.FsSlice{
|
{
|
||||||
{Gvk: resid.Gvk{Kind: "CustomResourceDefinition"}},
|
Gvk: resid.Gvk{Kind: "CustomResourceDefinition"},
|
||||||
{Gvk: resid.Gvk{Group: "apiregistration.k8s.io", Kind: "APIService"}},
|
},
|
||||||
{Gvk: resid.Gvk{Kind: "Namespace"}},
|
{
|
||||||
|
Gvk: resid.Gvk{Group: "apiregistration.k8s.io", Kind: "APIService"},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PrefixSuffixTransformerPlugin) Config(
|
func (p *PrefixSuffixTransformerPlugin) Config(
|
||||||
@@ -44,40 +48,45 @@ func (p *PrefixSuffixTransformerPlugin) Config(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *PrefixSuffixTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *PrefixSuffixTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
|
|
||||||
// Even if both the Prefix and Suffix are empty we want
|
// Even if both the Prefix and Suffix are empty we want
|
||||||
// to proceed with the transformation. This allows to add contextual
|
// to proceed with the transformation. This allows to add contextual
|
||||||
// information to the resources (AddNamePrefix and AddNameSuffix).
|
// information to the resources (AddNamePrefix and AddNameSuffix).
|
||||||
|
|
||||||
for _, r := range m.Resources() {
|
for _, r := range m.Resources() {
|
||||||
// TODO: move this test into the filter (i.e. make a better filter)
|
|
||||||
if p.shouldSkip(r.OrgId()) {
|
if p.shouldSkip(r.OrgId()) {
|
||||||
|
// Don't change the actual definition
|
||||||
|
// of a CRD.
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
id := r.OrgId()
|
id := r.OrgId()
|
||||||
// current default configuration contains
|
// current default configuration contains
|
||||||
// only one entry: "metadata/name" with no GVK
|
// only one entry: "metadata/name" with no GVK
|
||||||
for _, fs := range p.FieldSpecs {
|
for _, path := range p.FieldSpecs {
|
||||||
// TODO: this is redundant to filter (but needed for now)
|
if !id.IsSelected(&path.Gvk) {
|
||||||
if !id.IsSelected(&fs.Gvk) {
|
// With the currrent default configuration,
|
||||||
|
// because no Gvk is specified, so a wild
|
||||||
|
// card
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
// TODO: move this test into the filter.
|
|
||||||
if smellsLikeANameChange(&fs) {
|
if smellsLikeANameChange(&path) {
|
||||||
// "metadata/name" is the only field.
|
// "metadata/name" is the only field.
|
||||||
// this will add a prefix and a suffix
|
// this will add a prefix and a suffix
|
||||||
// to the resource even if those are
|
// to the resource even if those are
|
||||||
// empty
|
// empty
|
||||||
|
|
||||||
r.AddNamePrefix(p.Prefix)
|
r.AddNamePrefix(p.Prefix)
|
||||||
r.AddNameSuffix(p.Suffix)
|
r.AddNameSuffix(p.Suffix)
|
||||||
if p.Prefix != "" || p.Suffix != "" {
|
|
||||||
r.StorePreviousId()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
err := r.ApplyFilter(prefixsuffix.Filter{
|
|
||||||
Prefix: p.Prefix,
|
// the addPrefixSuffix method will not
|
||||||
Suffix: p.Suffix,
|
// change the name if both the prefix and suffix
|
||||||
FieldSpec: fs,
|
// are empty.
|
||||||
})
|
err := transform.MutateField(
|
||||||
|
r.Map(),
|
||||||
|
path.PathSlice(),
|
||||||
|
path.CreateIfNotPresent,
|
||||||
|
p.addPrefixSuffix)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -90,7 +99,8 @@ func smellsLikeANameChange(fs *types.FieldSpec) bool {
|
|||||||
return fs.Path == "metadata/name"
|
return fs.Path == "metadata/name"
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *PrefixSuffixTransformerPlugin) shouldSkip(id resid.ResId) bool {
|
func (p *PrefixSuffixTransformerPlugin) shouldSkip(
|
||||||
|
id resid.ResId) bool {
|
||||||
for _, path := range prefixSuffixFieldSpecsToSkip {
|
for _, path := range prefixSuffixFieldSpecsToSkip {
|
||||||
if id.IsSelected(&path.Gvk) {
|
if id.IsSelected(&path.Gvk) {
|
||||||
return true
|
return true
|
||||||
@@ -99,6 +109,15 @@ func (p *PrefixSuffixTransformerPlugin) shouldSkip(id resid.ResId) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *PrefixSuffixTransformerPlugin) addPrefixSuffix(
|
||||||
|
in interface{}) (interface{}, error) {
|
||||||
|
s, ok := in.(string)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("%#v is expected to be %T", in, s)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s%s%s", p.Prefix, s, p.Suffix), nil
|
||||||
|
}
|
||||||
|
|
||||||
func NewPrefixSuffixTransformerPlugin() resmap.TransformerPlugin {
|
func NewPrefixSuffixTransformerPlugin() resmap.TransformerPlugin {
|
||||||
return &PrefixSuffixTransformerPlugin{}
|
return &PrefixSuffixTransformerPlugin{}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,8 @@ package builtins
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/replicacount"
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
@@ -29,22 +30,18 @@ func (p *ReplicaCountTransformerPlugin) Config(
|
|||||||
|
|
||||||
func (p *ReplicaCountTransformerPlugin) Transform(m resmap.ResMap) error {
|
func (p *ReplicaCountTransformerPlugin) Transform(m resmap.ResMap) error {
|
||||||
found := false
|
found := false
|
||||||
for _, fs := range p.FieldSpecs {
|
for i, replicaSpec := range p.FieldSpecs {
|
||||||
matcher := p.createMatcher(fs)
|
matcher := p.createMatcher(i)
|
||||||
resList := m.GetMatchingResourcesByAnyId(matcher)
|
matchOriginal := m.GetMatchingResourcesByOriginalId(matcher)
|
||||||
if len(resList) > 0 {
|
matchCurrent := m.GetMatchingResourcesByCurrentId(matcher)
|
||||||
|
|
||||||
|
for _, res := range append(matchOriginal, matchCurrent...) {
|
||||||
found = true
|
found = true
|
||||||
for _, r := range resList {
|
err := transform.MutateField(
|
||||||
// There are redundant checks in the filter
|
res.Map(), replicaSpec.PathSlice(),
|
||||||
// that we'll live with until resolution of
|
replicaSpec.CreateIfNotPresent, p.addReplicas)
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/2506
|
if err != nil {
|
||||||
err := r.ApplyFilter(replicacount.Filter{
|
return err
|
||||||
Replica: p.Replica,
|
|
||||||
FieldSpec: fs,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -62,12 +59,30 @@ func (p *ReplicaCountTransformerPlugin) Transform(m resmap.ResMap) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Match Replica.Name and FieldSpec
|
// Match Replica.Name and FieldSpec
|
||||||
func (p *ReplicaCountTransformerPlugin) createMatcher(fs types.FieldSpec) resmap.IdMatcher {
|
func (p *ReplicaCountTransformerPlugin) createMatcher(i int) resmap.IdMatcher {
|
||||||
return func(r resid.ResId) bool {
|
return func(r resid.ResId) bool {
|
||||||
return r.Name == p.Replica.Name && r.Gvk.IsSelected(&fs.Gvk)
|
return r.Name == p.Replica.Name &&
|
||||||
|
r.Gvk.IsSelected(&p.FieldSpecs[i].Gvk)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *ReplicaCountTransformerPlugin) addReplicas(in interface{}) (interface{}, error) {
|
||||||
|
switch m := in.(type) {
|
||||||
|
case int64:
|
||||||
|
// Was already in the field.
|
||||||
|
case map[string]interface{}:
|
||||||
|
if len(m) != 0 {
|
||||||
|
// A map was already in the replicas field, don't want to
|
||||||
|
// discard this data silently.
|
||||||
|
return nil, fmt.Errorf("%#v is expected to be %T", in, m)
|
||||||
|
}
|
||||||
|
// Just got added, default type is map, but we can return anything.
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("%#v is expected to be %T", in, m)
|
||||||
|
}
|
||||||
|
return p.Replica.Count, nil
|
||||||
|
}
|
||||||
|
|
||||||
func NewReplicaCountTransformerPlugin() resmap.TransformerPlugin {
|
func NewReplicaCountTransformerPlugin() resmap.TransformerPlugin {
|
||||||
return &ReplicaCountTransformerPlugin{}
|
return &ReplicaCountTransformerPlugin{}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,10 +13,12 @@ import (
|
|||||||
type SecretGeneratorPlugin struct {
|
type SecretGeneratorPlugin struct {
|
||||||
h *resmap.PluginHelpers
|
h *resmap.PluginHelpers
|
||||||
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
types.ObjectMeta `json:"metadata,omitempty" yaml:"metadata,omitempty" protobuf:"bytes,1,opt,name=metadata"`
|
||||||
|
types.GeneratorOptions
|
||||||
types.SecretArgs
|
types.SecretArgs
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *SecretGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte) (err error) {
|
func (p *SecretGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte) (err error) {
|
||||||
|
p.GeneratorOptions = types.GeneratorOptions{}
|
||||||
p.SecretArgs = types.SecretArgs{}
|
p.SecretArgs = types.SecretArgs{}
|
||||||
err = yaml.Unmarshal(config, p)
|
err = yaml.Unmarshal(config, p)
|
||||||
if p.SecretArgs.Name == "" {
|
if p.SecretArgs.Name == "" {
|
||||||
@@ -31,7 +33,8 @@ func (p *SecretGeneratorPlugin) Config(h *resmap.PluginHelpers, config []byte) (
|
|||||||
|
|
||||||
func (p *SecretGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
func (p *SecretGeneratorPlugin) Generate() (resmap.ResMap, error) {
|
||||||
return p.h.ResmapFactory().FromSecretArgs(
|
return p.h.ResmapFactory().FromSecretArgs(
|
||||||
kv.NewLoader(p.h.Loader(), p.h.Validator()), p.SecretArgs)
|
kv.NewLoader(p.h.Loader(), p.h.Validator()),
|
||||||
|
&p.GeneratorOptions, p.SecretArgs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSecretGeneratorPlugin() resmap.GeneratorPlugin {
|
func NewSecretGeneratorPlugin() resmap.GeneratorPlugin {
|
||||||
|
|||||||
@@ -1,141 +0,0 @@
|
|||||||
// Code generated by pluginator on ValueAddTransformer; DO NOT EDIT.
|
|
||||||
// pluginator {unknown 1970-01-01T00:00:00Z }
|
|
||||||
|
|
||||||
package builtins
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/namespace"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/valueadd"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// An 'Add' transformer inspired by the IETF RFC 6902 JSON spec Add operation.
|
|
||||||
type ValueAddTransformerPlugin struct {
|
|
||||||
// Value is the value to add.
|
|
||||||
// Defaults to base name of encompassing kustomization root.
|
|
||||||
Value string `json:"value,omitempty" yaml:"value,omitempty"`
|
|
||||||
|
|
||||||
// Targets is a slice of targets that should have the value added.
|
|
||||||
Targets []Target `json:"targets,omitempty" yaml:"targets,omitempty"`
|
|
||||||
|
|
||||||
// TargetFilePath is a file path. If specified, the file will be parsed into
|
|
||||||
// a slice of Target, and appended to anything that was specified in the
|
|
||||||
// Targets field. This is just a means to share common target specifications.
|
|
||||||
TargetFilePath string `json:"targetFilePath,omitempty" yaml:"targetFilePath,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Target describes where to put the value.
|
|
||||||
type Target struct {
|
|
||||||
// Selector selects the resources to modify.
|
|
||||||
Selector *types.Selector `json:"selector,omitempty" yaml:"selector,omitempty"`
|
|
||||||
|
|
||||||
// NotSelector selects the resources to exclude
|
|
||||||
// from those included by overly broad selectors.
|
|
||||||
// TODO: implement this?
|
|
||||||
// NotSelector *types.Selector `json:"notSelector,omitempty" yaml:"notSelector,omitempty"`
|
|
||||||
|
|
||||||
// FieldPath is a JSON-style path to the field intended to hold the value.
|
|
||||||
FieldPath string `json:"fieldPath,omitempty" yaml:"fieldPath,omitempty"`
|
|
||||||
|
|
||||||
// FilePathPosition is passed to the filter directly. Look there for doc.
|
|
||||||
FilePathPosition int `json:"filePathPosition,omitempty" yaml:"filePathPosition,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *ValueAddTransformerPlugin) Config(h *resmap.PluginHelpers, c []byte) error {
|
|
||||||
err := yaml.Unmarshal(c, p)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Value = strings.TrimSpace(p.Value)
|
|
||||||
if p.Value == "" {
|
|
||||||
p.Value = filepath.Base(h.Loader().Root())
|
|
||||||
}
|
|
||||||
if p.TargetFilePath != "" {
|
|
||||||
bytes, err := h.Loader().Load(p.TargetFilePath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
var targets struct {
|
|
||||||
Targets []Target `json:"targets,omitempty" yaml:"targets,omitempty"`
|
|
||||||
}
|
|
||||||
err = yaml.Unmarshal(bytes, &targets)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
p.Targets = append(p.Targets, targets.Targets...)
|
|
||||||
}
|
|
||||||
if len(p.Targets) == 0 {
|
|
||||||
return fmt.Errorf("must specify at least one target")
|
|
||||||
}
|
|
||||||
for _, target := range p.Targets {
|
|
||||||
if err = validateSelector(target.Selector); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
// TODO: call validateSelector(target.NotSelector) if field added.
|
|
||||||
if err = validateJsonFieldPath(target.FieldPath); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if target.FilePathPosition < 0 {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"value of FilePathPosition (%d) cannot be negative",
|
|
||||||
target.FilePathPosition)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: implement
|
|
||||||
func validateSelector(_ *types.Selector) error {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Enforce RFC 6902?
|
|
||||||
func validateJsonFieldPath(p string) error {
|
|
||||||
if len(p) == 0 {
|
|
||||||
return fmt.Errorf("fieldPath cannot be empty")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *ValueAddTransformerPlugin) Transform(m resmap.ResMap) (err error) {
|
|
||||||
for _, t := range p.Targets {
|
|
||||||
var resources []*resource.Resource
|
|
||||||
if t.Selector == nil {
|
|
||||||
resources = m.Resources()
|
|
||||||
} else {
|
|
||||||
resources, err = m.Select(*t.Selector)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO: consider t.NotSelector if implemented
|
|
||||||
for _, res := range resources {
|
|
||||||
if t.FieldPath == types.MetadataNamespacePath {
|
|
||||||
err = res.ApplyFilter(namespace.Filter{
|
|
||||||
Namespace: p.Value,
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
err = res.ApplyFilter(valueadd.Filter{
|
|
||||||
Value: p.Value,
|
|
||||||
FieldPath: t.FieldPath,
|
|
||||||
FilePathPosition: t.FilePathPosition,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewValueAddTransformerPlugin() resmap.TransformerPlugin {
|
|
||||||
return &ValueAddTransformerPlugin{}
|
|
||||||
}
|
|
||||||
@@ -4,7 +4,6 @@
|
|||||||
package filesys_test
|
package filesys_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
@@ -99,7 +98,6 @@ func TestNewTempConfirmDir(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
defer os.RemoveAll(string(tmp))
|
|
||||||
|
|
||||||
delinked, err := filepath.EvalSymlinks(string(tmp))
|
delinked, err := filepath.EvalSymlinks(string(tmp))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -3,11 +3,7 @@
|
|||||||
|
|
||||||
package filesys
|
package filesys
|
||||||
|
|
||||||
import (
|
import "path/filepath"
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
// RootedPath returns a rooted path, e.g. "/foo/bar" as
|
// RootedPath returns a rooted path, e.g. "/foo/bar" as
|
||||||
// opposed to "foo/bar".
|
// opposed to "foo/bar".
|
||||||
@@ -32,94 +28,3 @@ func StripLeadingSeps(s string) string {
|
|||||||
}
|
}
|
||||||
return s[k:]
|
return s[k:]
|
||||||
}
|
}
|
||||||
|
|
||||||
// PathSplit converts a file path to a slice of string.
|
|
||||||
// If the path is absolute (if the path has a leading slash),
|
|
||||||
// then the first entry in the result is an empty string.
|
|
||||||
// Desired: path == PathJoin(PathSplit(path))
|
|
||||||
func PathSplit(incoming string) []string {
|
|
||||||
if incoming == "" {
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
dir, path := filepath.Split(incoming)
|
|
||||||
if dir == string(os.PathSeparator) {
|
|
||||||
if path == "" {
|
|
||||||
return []string{""}
|
|
||||||
}
|
|
||||||
return []string{"", path}
|
|
||||||
}
|
|
||||||
dir = strings.TrimSuffix(dir, string(os.PathSeparator))
|
|
||||||
if dir == "" {
|
|
||||||
return []string{path}
|
|
||||||
}
|
|
||||||
return append(PathSplit(dir), path)
|
|
||||||
}
|
|
||||||
|
|
||||||
// PathJoin converts a slice of string to a file path.
|
|
||||||
// If the first entry is an empty string, then the returned
|
|
||||||
// path is absolute (it has a leading slash).
|
|
||||||
// Desired: path == PathJoin(PathSplit(path))
|
|
||||||
func PathJoin(incoming []string) string {
|
|
||||||
if len(incoming) == 0 {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
if incoming[0] == "" {
|
|
||||||
return string(os.PathSeparator) + filepath.Join(incoming[1:]...)
|
|
||||||
}
|
|
||||||
return filepath.Join(incoming...)
|
|
||||||
}
|
|
||||||
|
|
||||||
// InsertPathPart inserts 'part' at position 'pos' in the given filepath.
|
|
||||||
// The first position is 0.
|
|
||||||
//
|
|
||||||
// E.g. if part == 'PEACH'
|
|
||||||
//
|
|
||||||
// OLD : NEW : POS
|
|
||||||
// --------------------------------------------------------
|
|
||||||
// {empty} : PEACH : irrelevant
|
|
||||||
// / : /PEACH : irrelevant
|
|
||||||
// pie : PEACH/pie : 0 (or negative)
|
|
||||||
// /pie : /PEACH/pie : 0 (or negative)
|
|
||||||
// raw : raw/PEACH : 1 (or larger)
|
|
||||||
// /raw : /raw/PEACH : 1 (or larger)
|
|
||||||
// a/nice/warm/pie : a/nice/warm/PEACH/pie : 3
|
|
||||||
// /a/nice/warm/pie : /a/nice/warm/PEACH/pie : 3
|
|
||||||
//
|
|
||||||
// * An empty part results in no change.
|
|
||||||
//
|
|
||||||
// * Absolute paths get their leading '/' stripped, treated like
|
|
||||||
// relative paths, and the leading '/' is re-added on output.
|
|
||||||
// The meaning of pos is intentionally the same in either absolute or
|
|
||||||
// relative paths; if it weren't, this function could convert absolute
|
|
||||||
// paths to relative paths, which is not desirable.
|
|
||||||
//
|
|
||||||
// * For robustness (liberal input, conservative output) Pos values that
|
|
||||||
// that are too small (large) to index the split filepath result in a
|
|
||||||
// prefix (postfix) rather than an error. Use extreme position values
|
|
||||||
// to assure a prefix or postfix (e.g. 0 will always prefix, and
|
|
||||||
// 9999 will presumably always postfix).
|
|
||||||
func InsertPathPart(path string, pos int, part string) string {
|
|
||||||
if part == "" {
|
|
||||||
return path
|
|
||||||
}
|
|
||||||
parts := PathSplit(path)
|
|
||||||
if pos < 0 {
|
|
||||||
pos = 0
|
|
||||||
} else if pos > len(parts) {
|
|
||||||
pos = len(parts)
|
|
||||||
}
|
|
||||||
if len(parts) > 0 && parts[0] == "" && pos < len(parts) {
|
|
||||||
// An empty string at 0 indicates an absolute path, and means
|
|
||||||
// we must increment pos. This change means that a position
|
|
||||||
// specification has the same meaning in relative and absolute paths.
|
|
||||||
// E.g. in either the path 'a/b/c' or the path '/a/b/c',
|
|
||||||
// 'a' is at 0, 'b' is at 1 and 'c' is at 2, and inserting at
|
|
||||||
// zero means a new first field _without_ changing an absolute
|
|
||||||
// path to a relative path.
|
|
||||||
pos++
|
|
||||||
}
|
|
||||||
result := make([]string, len(parts)+1)
|
|
||||||
copy(result, parts[0:pos])
|
|
||||||
result[pos] = part
|
|
||||||
return PathJoin(append(result, parts[pos:]...))
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
package filesys_test
|
package filesys_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
@@ -94,11 +93,6 @@ func TestFilePathSplit(t *testing.T) {
|
|||||||
dir: "",
|
dir: "",
|
||||||
file: "rabbit.jpg",
|
file: "rabbit.jpg",
|
||||||
},
|
},
|
||||||
{
|
|
||||||
full: "/",
|
|
||||||
dir: "/",
|
|
||||||
file: "",
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
full: "/beans",
|
full: "/beans",
|
||||||
dir: "/",
|
dir: "/",
|
||||||
@@ -130,169 +124,6 @@ func TestFilePathSplit(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPathSplitAndJoin(t *testing.T) {
|
|
||||||
cases := map[string]struct {
|
|
||||||
original string
|
|
||||||
expected []string
|
|
||||||
}{
|
|
||||||
"Empty": {
|
|
||||||
original: "",
|
|
||||||
expected: []string{},
|
|
||||||
},
|
|
||||||
"One": {
|
|
||||||
original: "hello",
|
|
||||||
expected: []string{"hello"},
|
|
||||||
},
|
|
||||||
"Two": {
|
|
||||||
original: "hello/there",
|
|
||||||
expected: []string{"hello", "there"},
|
|
||||||
},
|
|
||||||
"Three": {
|
|
||||||
original: "hello/my/friend",
|
|
||||||
expected: []string{"hello", "my", "friend"},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
for n, c := range cases {
|
|
||||||
f := func(t *testing.T, original string, expected []string) {
|
|
||||||
actual := PathSplit(original)
|
|
||||||
if len(actual) != len(expected) {
|
|
||||||
t.Fatalf(
|
|
||||||
"expected len %d, got len %d",
|
|
||||||
len(expected), len(actual))
|
|
||||||
}
|
|
||||||
for i := range expected {
|
|
||||||
if expected[i] != actual[i] {
|
|
||||||
t.Fatalf(
|
|
||||||
"at i=%d, expected '%s', got '%s'",
|
|
||||||
i, expected[i], actual[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
joined := PathJoin(actual)
|
|
||||||
if joined != original {
|
|
||||||
t.Fatalf(
|
|
||||||
"when rejoining, expected '%s', got '%s'",
|
|
||||||
original, joined)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
t.Run("relative"+n, func(t *testing.T) {
|
|
||||||
f(t, c.original, c.expected)
|
|
||||||
})
|
|
||||||
t.Run("absolute"+n, func(t *testing.T) {
|
|
||||||
f(t,
|
|
||||||
string(os.PathSeparator)+c.original,
|
|
||||||
append([]string{""}, c.expected...))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestInsertPathPart(t *testing.T) {
|
|
||||||
cases := map[string]struct {
|
|
||||||
original string
|
|
||||||
pos int
|
|
||||||
part string
|
|
||||||
expected string
|
|
||||||
}{
|
|
||||||
"rootOne": {
|
|
||||||
original: "/",
|
|
||||||
pos: 0,
|
|
||||||
part: "___",
|
|
||||||
expected: "/___",
|
|
||||||
},
|
|
||||||
"rootTwo": {
|
|
||||||
original: "/",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "/___",
|
|
||||||
},
|
|
||||||
"rootedFirst": {
|
|
||||||
original: "/apple",
|
|
||||||
pos: 0,
|
|
||||||
part: "___",
|
|
||||||
expected: "/___/apple",
|
|
||||||
},
|
|
||||||
"rootedSecond": {
|
|
||||||
original: "/apple",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "/apple/___",
|
|
||||||
},
|
|
||||||
"rootedThird": {
|
|
||||||
original: "/apple/banana",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "/apple/banana/___",
|
|
||||||
},
|
|
||||||
"emptyLow": {
|
|
||||||
original: "",
|
|
||||||
pos: -3,
|
|
||||||
part: "___",
|
|
||||||
expected: "___",
|
|
||||||
},
|
|
||||||
"emptyHigh": {
|
|
||||||
original: "",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "___",
|
|
||||||
},
|
|
||||||
"peachPie": {
|
|
||||||
original: "a/nice/warm/pie",
|
|
||||||
pos: 3,
|
|
||||||
part: "PEACH",
|
|
||||||
expected: "a/nice/warm/PEACH/pie",
|
|
||||||
},
|
|
||||||
"rootedPeachPie": {
|
|
||||||
original: "/a/nice/warm/pie",
|
|
||||||
pos: 3,
|
|
||||||
part: "PEACH",
|
|
||||||
expected: "/a/nice/warm/PEACH/pie",
|
|
||||||
},
|
|
||||||
"longStart": {
|
|
||||||
original: "a/b/c/d/e/f",
|
|
||||||
pos: 0,
|
|
||||||
part: "___",
|
|
||||||
expected: "___/a/b/c/d/e/f",
|
|
||||||
},
|
|
||||||
"rootedLongStart": {
|
|
||||||
original: "/a/b/c/d/e/f",
|
|
||||||
pos: 0,
|
|
||||||
part: "___",
|
|
||||||
expected: "/___/a/b/c/d/e/f",
|
|
||||||
},
|
|
||||||
"longMiddle": {
|
|
||||||
original: "a/b/c/d/e/f",
|
|
||||||
pos: 3,
|
|
||||||
part: "___",
|
|
||||||
expected: "a/b/c/___/d/e/f",
|
|
||||||
},
|
|
||||||
"rootedLongMiddle": {
|
|
||||||
original: "/a/b/c/d/e/f",
|
|
||||||
pos: 3,
|
|
||||||
part: "___",
|
|
||||||
expected: "/a/b/c/___/d/e/f",
|
|
||||||
},
|
|
||||||
"longEnd": {
|
|
||||||
original: "a/b/c/d/e/f",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "a/b/c/d/e/f/___",
|
|
||||||
},
|
|
||||||
"rootedLongEnd": {
|
|
||||||
original: "/a/b/c/d/e/f",
|
|
||||||
pos: 444,
|
|
||||||
part: "___",
|
|
||||||
expected: "/a/b/c/d/e/f/___",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
for n, c := range cases {
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
actual := InsertPathPart(c.original, c.pos, c.part)
|
|
||||||
if actual != c.expected {
|
|
||||||
t.Fatalf("expected '%s', got '%s'", c.expected, actual)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestStripTrailingSeps(t *testing.T) {
|
func TestStripTrailingSeps(t *testing.T) {
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
full string
|
full string
|
||||||
|
|||||||
@@ -1,44 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package annotations
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type annoMap map[string]string
|
|
||||||
|
|
||||||
type Filter struct {
|
|
||||||
// Annotations is the set of annotations to apply to the inputs
|
|
||||||
Annotations annoMap `yaml:"annotations,omitempty"`
|
|
||||||
|
|
||||||
// FsSlice contains the FieldSpecs to locate the namespace field
|
|
||||||
FsSlice types.FsSlice
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
keys := yaml.SortedMapKeys(f.Annotations)
|
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(
|
|
||||||
func(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
for _, k := range keys {
|
|
||||||
if err := node.PipeE(fsslice.Filter{
|
|
||||||
FsSlice: f.FsSlice,
|
|
||||||
SetValue: filtersutil.SetEntry(
|
|
||||||
k, f.Annotations[k], yaml.NodeTagString),
|
|
||||||
CreateKind: yaml.MappingNode, // Annotations are MappingNodes.
|
|
||||||
CreateTag: yaml.NodeTagMap,
|
|
||||||
}); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
})).Filter(nodes)
|
|
||||||
return nodes, err
|
|
||||||
}
|
|
||||||
@@ -1,226 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package annotations
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var annosFs = builtinconfig.MakeDefaultConfig().CommonAnnotations
|
|
||||||
|
|
||||||
func TestAnnotations_Filter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedOutput string
|
|
||||||
filter Filter
|
|
||||||
fsslice types.FsSlice
|
|
||||||
}{
|
|
||||||
"add": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
auto: ford
|
|
||||||
bean: cannellini
|
|
||||||
clown: emmett kelley
|
|
||||||
dragon: smaug
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"clown": "emmett kelley",
|
|
||||||
"auto": "ford",
|
|
||||||
"dragon": "smaug",
|
|
||||||
"bean": "cannellini",
|
|
||||||
}},
|
|
||||||
},
|
|
||||||
"update": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: superman
|
|
||||||
fiend: luthor
|
|
||||||
bean: cannellini
|
|
||||||
clown: emmett kelley
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"clown": "emmett kelley",
|
|
||||||
"hero": "superman",
|
|
||||||
"fiend": "luthor",
|
|
||||||
"bean": "cannellini",
|
|
||||||
}},
|
|
||||||
},
|
|
||||||
"data-fieldspecs": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
sleater: kinney
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
sleater: kinney
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
sleater: kinney
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
sleater: kinney
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"sleater": "kinney",
|
|
||||||
}},
|
|
||||||
fsslice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "a/b",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"number": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
2: ford
|
|
||||||
clown: "1"
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"clown": "1",
|
|
||||||
"2": "ford",
|
|
||||||
}},
|
|
||||||
},
|
|
||||||
|
|
||||||
// test quoting of values which are not considered strings in yaml 1.1
|
|
||||||
"yaml_1_1_compatibility": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
a: "y"
|
|
||||||
b: y1
|
|
||||||
c: "yes"
|
|
||||||
d: yes1
|
|
||||||
e: "true"
|
|
||||||
f: true1
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"a": "y",
|
|
||||||
"b": "y1",
|
|
||||||
"c": "yes",
|
|
||||||
"d": "yes1",
|
|
||||||
"e": "true",
|
|
||||||
"f": "true1",
|
|
||||||
}},
|
|
||||||
},
|
|
||||||
|
|
||||||
// test quoting of values which are not considered strings in yaml 1.1
|
|
||||||
"null_annotations": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations: null
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
annotations:
|
|
||||||
a: a1
|
|
||||||
b: b1
|
|
||||||
`,
|
|
||||||
filter: Filter{Annotations: annoMap{
|
|
||||||
"a": "a1",
|
|
||||||
"b": "b1",
|
|
||||||
}},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
filter := tc.filter
|
|
||||||
filter.FsSlice = append(annosFs, tc.fsslice...)
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(filtertest_test.RunFilter(t, tc.input, filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package annotations contains a kio.Filter implementation of the kustomize
|
|
||||||
// annotations transformer.
|
|
||||||
package annotations
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package annotations
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
fss := builtinconfig.MakeDefaultConfig().CommonAnnotations
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{Filter{
|
|
||||||
Annotations: map[string]string{
|
|
||||||
"foo": "bar",
|
|
||||||
"booleanValue": "true",
|
|
||||||
"numberValue": "42",
|
|
||||||
},
|
|
||||||
FsSlice: fss,
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// annotations:
|
|
||||||
// booleanValue: "true"
|
|
||||||
// foo: bar
|
|
||||||
// numberValue: "42"
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// annotations:
|
|
||||||
// booleanValue: "true"
|
|
||||||
// foo: bar
|
|
||||||
// numberValue: "42"
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package fieldspec contains a yaml.Filter to modify a resource
|
|
||||||
// that matches the FieldSpec.
|
|
||||||
package fieldspec
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fieldspec_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
. "sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
in := &kio.ByteReader{
|
|
||||||
Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`),
|
|
||||||
}
|
|
||||||
fltr := Filter{
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
SetValue: filtersutil.SetScalar("green"),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "a/b", CreateIfNotPresent: true},
|
|
||||||
}
|
|
||||||
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{in},
|
|
||||||
Filters: []kio.Filter{kio.FilterAll(fltr)},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// a:
|
|
||||||
// b: green
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// a:
|
|
||||||
// b: green
|
|
||||||
}
|
|
||||||
@@ -1,185 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fieldspec
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/utils"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/errors"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
var _ yaml.Filter = Filter{}
|
|
||||||
|
|
||||||
// Filter possibly mutates its object argument using a FieldSpec.
|
|
||||||
// If the object matches the FieldSpec, and the node found
|
|
||||||
// by following the fieldSpec's path is non-null, this filter calls
|
|
||||||
// the setValue function on the node at the end of the path.
|
|
||||||
// If any part of the path doesn't exist, the filter returns
|
|
||||||
// without doing anything and without error, unless it was set
|
|
||||||
// to create the path. If set to create, it creates a tree of maps
|
|
||||||
// along the path, and the leaf node gets the setValue called on it.
|
|
||||||
// Error on GVK mismatch, empty or poorly formed path.
|
|
||||||
// Filter expect kustomize style paths, not JSON paths.
|
|
||||||
// Filter stores internal state and should not be reused
|
|
||||||
type Filter struct {
|
|
||||||
// FieldSpec contains the path to the value to set.
|
|
||||||
FieldSpec types.FieldSpec `yaml:"fieldSpec"`
|
|
||||||
|
|
||||||
// Set the field using this function
|
|
||||||
SetValue filtersutil.SetFn
|
|
||||||
|
|
||||||
// CreateKind defines the type of node to create if the field is not found
|
|
||||||
CreateKind yaml.Kind
|
|
||||||
|
|
||||||
CreateTag string
|
|
||||||
|
|
||||||
// path keeps internal state about the current path
|
|
||||||
path []string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (fltr Filter) Filter(obj *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
// check if the FieldSpec applies to the object
|
|
||||||
if match, err := isMatchGVK(fltr.FieldSpec, obj); !match || err != nil {
|
|
||||||
return obj, errors.Wrap(err)
|
|
||||||
}
|
|
||||||
fltr.path = utils.PathSplitter(fltr.FieldSpec.Path)
|
|
||||||
err := fltr.filter(obj)
|
|
||||||
if err != nil {
|
|
||||||
s, _ := obj.String()
|
|
||||||
return nil, errors.WrapPrefixf(err,
|
|
||||||
"considering field '%s' of object\n%v", fltr.FieldSpec.Path, s)
|
|
||||||
}
|
|
||||||
return obj, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Recursively called.
|
|
||||||
func (fltr Filter) filter(obj *yaml.RNode) error {
|
|
||||||
if len(fltr.path) == 0 {
|
|
||||||
// found the field -- set its value
|
|
||||||
return fltr.SetValue(obj)
|
|
||||||
}
|
|
||||||
if obj.IsTaggedNull() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
switch obj.YNode().Kind {
|
|
||||||
case yaml.SequenceNode:
|
|
||||||
return fltr.handleSequence(obj)
|
|
||||||
case yaml.MappingNode:
|
|
||||||
return fltr.handleMap(obj)
|
|
||||||
case yaml.AliasNode:
|
|
||||||
return fltr.filter(yaml.NewRNode(obj.YNode().Alias))
|
|
||||||
default:
|
|
||||||
return errors.Errorf("expected sequence or mapping node")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// handleMap calls filter on the map field matching the next path element
|
|
||||||
func (fltr Filter) handleMap(obj *yaml.RNode) error {
|
|
||||||
fieldName, isSeq := isSequenceField(fltr.path[0])
|
|
||||||
if fieldName == "" {
|
|
||||||
return fmt.Errorf("cannot set or create an empty field name")
|
|
||||||
}
|
|
||||||
// lookup the field matching the next path element
|
|
||||||
var operation yaml.Filter
|
|
||||||
var kind yaml.Kind
|
|
||||||
tag := yaml.NodeTagEmpty
|
|
||||||
switch {
|
|
||||||
case !fltr.FieldSpec.CreateIfNotPresent || fltr.CreateKind == 0 || isSeq:
|
|
||||||
// don't create the field if we don't find it
|
|
||||||
operation = yaml.Lookup(fieldName)
|
|
||||||
if isSeq {
|
|
||||||
// The query path thinks this field should be a sequence;
|
|
||||||
// accept this hint for use later if the tag is NodeTagNull.
|
|
||||||
kind = yaml.SequenceNode
|
|
||||||
}
|
|
||||||
case len(fltr.path) <= 1:
|
|
||||||
// create the field if it is missing: use the provided node kind
|
|
||||||
operation = yaml.LookupCreate(fltr.CreateKind, fieldName)
|
|
||||||
kind = fltr.CreateKind
|
|
||||||
tag = fltr.CreateTag
|
|
||||||
default:
|
|
||||||
// create the field if it is missing: must be a mapping node
|
|
||||||
operation = yaml.LookupCreate(yaml.MappingNode, fieldName)
|
|
||||||
kind = yaml.MappingNode
|
|
||||||
tag = yaml.NodeTagMap
|
|
||||||
}
|
|
||||||
|
|
||||||
// locate (or maybe create) the field
|
|
||||||
field, err := obj.Pipe(operation)
|
|
||||||
if err != nil {
|
|
||||||
return errors.WrapPrefixf(err, "fieldName: %s", fieldName)
|
|
||||||
}
|
|
||||||
if field == nil {
|
|
||||||
// No error if field not found.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the value exists, but is null and kind is set,
|
|
||||||
// then change it to the creation type
|
|
||||||
// TODO: update yaml.LookupCreate to support this
|
|
||||||
if field.YNode().Tag == yaml.NodeTagNull && yaml.IsCreate(kind) {
|
|
||||||
field.YNode().Kind = kind
|
|
||||||
field.YNode().Tag = tag
|
|
||||||
}
|
|
||||||
|
|
||||||
// copy the current fltr and change the path on the copy
|
|
||||||
var next = fltr
|
|
||||||
// call filter for the next path element on the matching field
|
|
||||||
next.path = fltr.path[1:]
|
|
||||||
return next.filter(field)
|
|
||||||
}
|
|
||||||
|
|
||||||
// seq calls filter on all sequence elements
|
|
||||||
func (fltr Filter) handleSequence(obj *yaml.RNode) error {
|
|
||||||
if err := obj.VisitElements(func(node *yaml.RNode) error {
|
|
||||||
// recurse on each element -- re-allocating a Filter is
|
|
||||||
// not strictly required, but is more consistent with field
|
|
||||||
// and less likely to have side effects
|
|
||||||
// keep the entire path -- it does not contain parts for sequences
|
|
||||||
return fltr.filter(node)
|
|
||||||
}); err != nil {
|
|
||||||
return errors.WrapPrefixf(err,
|
|
||||||
"visit traversal on path: %v", fltr.path)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// isSequenceField returns true if the path element is for a sequence field.
|
|
||||||
// isSequence also returns the path element with the '[]' suffix trimmed
|
|
||||||
func isSequenceField(name string) (string, bool) {
|
|
||||||
shorter := strings.TrimSuffix(name, "[]")
|
|
||||||
return shorter, shorter != name
|
|
||||||
}
|
|
||||||
|
|
||||||
// isMatchGVK returns true if the fs.GVK matches the obj GVK.
|
|
||||||
func isMatchGVK(fs types.FieldSpec, obj *yaml.RNode) (bool, error) {
|
|
||||||
meta, err := obj.GetMeta()
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
if fs.Kind != "" && fs.Kind != meta.Kind {
|
|
||||||
// kind doesn't match
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// parse the group and version from the apiVersion field
|
|
||||||
group, version := parseGV(meta.APIVersion)
|
|
||||||
|
|
||||||
if fs.Group != "" && fs.Group != group {
|
|
||||||
// group doesn't match
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if fs.Version != "" && fs.Version != version {
|
|
||||||
// version doesn't match
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
@@ -1,564 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fieldspec_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestFilter_Filter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter fieldspec.Filter
|
|
||||||
fieldSpec string
|
|
||||||
error string
|
|
||||||
}{
|
|
||||||
"path not found": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"empty path": {
|
|
||||||
fieldSpec: `
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
`,
|
|
||||||
error: `considering field '' of object
|
|
||||||
apiVersion: foo
|
|
||||||
kind: Bar
|
|
||||||
xxx:
|
|
||||||
: cannot set or create an empty field name`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: e
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update-kind-not-match": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
kind: Bar1
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar2
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar2
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update-group-not-match": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo2/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo2/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update-version-not-match": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
version: v1beta1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta2
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta2
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"bad-version": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
version: v1beta1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta2/something
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta2/something
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"bad-meta": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: foo
|
|
||||||
version: v1beta1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
a:
|
|
||||||
b: c
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
error: "missing Resource metadata",
|
|
||||||
},
|
|
||||||
|
|
||||||
"miss-match-type": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b/c
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: a
|
|
||||||
`,
|
|
||||||
error: `considering field 'a/b/c' of object
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: a
|
|
||||||
: expected sequence or mapping node`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"add": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b/c/d
|
|
||||||
group: foo
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a: {}
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a: {b: {c: {d: e}}}
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"update-in-sequence": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b[]/c/d
|
|
||||||
group: foo
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
- c:
|
|
||||||
d: a
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
- c:
|
|
||||||
d: e
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
// Don't create a sequence
|
|
||||||
"empty-sequence-no-create": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b[]/c/d
|
|
||||||
group: foo
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a: {}
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a: {}
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
// Create a new field for an element in a sequence
|
|
||||||
"empty-sequence-create": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b[]/c/d
|
|
||||||
group: foo
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
- c: {}
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1beta1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
- c: {d: e}
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"group v1": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
group: v1
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"version v1": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b
|
|
||||||
version: v1
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b: e
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"successfully set field on array entry no sequence hint": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: spec/containers/image
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: bar
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"successfully set field on array entry with sequence hint": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: spec/containers[]/image
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: bar
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"failure to set field on array entry with sequence hint in path": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: spec/containers[]/image
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers: []
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"failure to set field on array entry, no sequence hint in path": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: spec/containers/image
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"fieldname with slash '/'": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b\/c/d
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b/c:
|
|
||||||
d: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b/c:
|
|
||||||
d: bar
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"fieldname with multiple '/'": {
|
|
||||||
fieldSpec: `
|
|
||||||
path: a/b\/c/d\/e/f
|
|
||||||
version: v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b/c:
|
|
||||||
d/e:
|
|
||||||
f: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Bar
|
|
||||||
a:
|
|
||||||
b/c:
|
|
||||||
d/e:
|
|
||||||
f: bar
|
|
||||||
`,
|
|
||||||
filter: fieldspec.Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("bar"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for n := range testCases {
|
|
||||||
tc := testCases[n]
|
|
||||||
t.Run(n, func(t *testing.T) {
|
|
||||||
err := yaml.Unmarshal([]byte(tc.fieldSpec), &tc.filter.FieldSpec)
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
|
|
||||||
out := &bytes.Buffer{}
|
|
||||||
rw := &kio.ByteReadWriter{
|
|
||||||
Reader: bytes.NewBufferString(tc.input),
|
|
||||||
Writer: out,
|
|
||||||
OmitReaderAnnotations: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
// run the filter
|
|
||||||
err = kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{rw},
|
|
||||||
Filters: []kio.Filter{kio.FilterAll(tc.filter)},
|
|
||||||
Outputs: []kio.Writer{rw},
|
|
||||||
}.Execute()
|
|
||||||
if tc.error != "" {
|
|
||||||
if !assert.EqualError(t, err, tc.error) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
// stop rest of test
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
|
|
||||||
// check results
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expected),
|
|
||||||
strings.TrimSpace(out.String())) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
package fieldspec
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Return true for 'v' followed by a 1 or 2, and don't look at rest.
|
|
||||||
// I.e. 'v1', 'v1beta1', 'v2', would return true.
|
|
||||||
func looksLikeACoreApiVersion(s string) bool {
|
|
||||||
if len(s) < 2 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if s[0:1] != "v" {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return s[1:2] == "1" || s[1:2] == "2"
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseGV parses apiVersion field into group and version.
|
|
||||||
func parseGV(apiVersion string) (group, version string) {
|
|
||||||
// parse the group and version from the apiVersion field
|
|
||||||
parts := strings.SplitN(apiVersion, "/", 2)
|
|
||||||
group = parts[0]
|
|
||||||
if len(parts) > 1 {
|
|
||||||
version = parts[1]
|
|
||||||
}
|
|
||||||
// Special case the original "apiVersion" of what
|
|
||||||
// we now call the "core" (empty) group.
|
|
||||||
if version == "" && looksLikeACoreApiVersion(group) {
|
|
||||||
version = group
|
|
||||||
group = ""
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetGVK parses the metadata into a GVK
|
|
||||||
func GetGVK(meta yaml.ResourceMeta) resid.Gvk {
|
|
||||||
group, version := parseGV(meta.APIVersion)
|
|
||||||
return resid.Gvk{
|
|
||||||
Group: group,
|
|
||||||
Version: version,
|
|
||||||
Kind: meta.Kind,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,156 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
package fieldspec
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestParseGV(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedGroup string
|
|
||||||
expectedVersion string
|
|
||||||
}{
|
|
||||||
"empty": {
|
|
||||||
input: "",
|
|
||||||
expectedGroup: "",
|
|
||||||
expectedVersion: "",
|
|
||||||
},
|
|
||||||
"certSigning": {
|
|
||||||
input: "certificates.k8s.io/v1beta1",
|
|
||||||
expectedGroup: "certificates.k8s.io",
|
|
||||||
expectedVersion: "v1beta1",
|
|
||||||
},
|
|
||||||
"extensions": {
|
|
||||||
input: "extensions/v1beta1",
|
|
||||||
expectedGroup: "extensions",
|
|
||||||
expectedVersion: "v1beta1",
|
|
||||||
},
|
|
||||||
"normal": {
|
|
||||||
input: "apps/v1",
|
|
||||||
expectedGroup: "apps",
|
|
||||||
expectedVersion: "v1",
|
|
||||||
},
|
|
||||||
"justApps": {
|
|
||||||
input: "apps",
|
|
||||||
expectedGroup: "apps",
|
|
||||||
expectedVersion: "",
|
|
||||||
},
|
|
||||||
"coreV1": {
|
|
||||||
input: "v1",
|
|
||||||
expectedGroup: "",
|
|
||||||
expectedVersion: "v1",
|
|
||||||
},
|
|
||||||
"coreV2": {
|
|
||||||
input: "v2",
|
|
||||||
expectedGroup: "",
|
|
||||||
expectedVersion: "v2",
|
|
||||||
},
|
|
||||||
"coreV2Beta1": {
|
|
||||||
input: "v2beta1",
|
|
||||||
expectedGroup: "",
|
|
||||||
expectedVersion: "v2beta1",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
group, version := parseGV(tc.input)
|
|
||||||
if !assert.Equal(t, tc.expectedGroup, group) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
if !assert.Equal(t, tc.expectedVersion, version) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetGVK(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expected resid.Gvk
|
|
||||||
parseError string
|
|
||||||
metaError string
|
|
||||||
}{
|
|
||||||
"empty": {
|
|
||||||
input: `
|
|
||||||
`,
|
|
||||||
parseError: "EOF",
|
|
||||||
},
|
|
||||||
"junk": {
|
|
||||||
input: `
|
|
||||||
congress: effective
|
|
||||||
`,
|
|
||||||
metaError: "missing Resource metadata",
|
|
||||||
},
|
|
||||||
"normal": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
`,
|
|
||||||
expected: resid.Gvk{Group: "apps", Version: "v1", Kind: "Deployment"},
|
|
||||||
},
|
|
||||||
"apiVersionOnlyWithSlash": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
`,
|
|
||||||
expected: resid.Gvk{Group: "apps", Version: "v1", Kind: ""},
|
|
||||||
},
|
|
||||||
"apiVersionOnlyNoSlash1": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps
|
|
||||||
`,
|
|
||||||
expected: resid.Gvk{Group: "apps", Version: "", Kind: ""},
|
|
||||||
},
|
|
||||||
"apiVersionOnlyNoSlash2": {
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
`,
|
|
||||||
expected: resid.Gvk{Group: "", Version: "v1", Kind: ""},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
obj, err := yaml.Parse(tc.input)
|
|
||||||
if len(tc.parseError) != 0 {
|
|
||||||
if err == nil {
|
|
||||||
t.Error("expected parse error")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !strings.Contains(err.Error(), tc.parseError) {
|
|
||||||
t.Errorf("expected parse err '%s', got '%v'", tc.parseError, err)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
meta, err := obj.GetMeta()
|
|
||||||
if len(tc.metaError) != 0 {
|
|
||||||
if err == nil {
|
|
||||||
t.Error("expected meta error")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !strings.Contains(err.Error(), tc.metaError) {
|
|
||||||
t.Errorf("expected meta err '%s', got '%v'", tc.metaError, err)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
gvk := GetGVK(meta)
|
|
||||||
if !assert.Equal(t, tc.expected, gvk) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,33 +0,0 @@
|
|||||||
package filtersutil
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SetFn is a function that accepts an RNode to possibly modify.
|
|
||||||
type SetFn func(*yaml.RNode) error
|
|
||||||
|
|
||||||
// SetScalar returns a SetFn to set a scalar value
|
|
||||||
func SetScalar(value string) SetFn {
|
|
||||||
return func(node *yaml.RNode) error {
|
|
||||||
return node.PipeE(yaml.FieldSetter{StringValue: value})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetEntry returns a SetFn to set an entry in a map
|
|
||||||
func SetEntry(key, value, tag string) SetFn {
|
|
||||||
n := &yaml.Node{
|
|
||||||
Kind: yaml.ScalarNode,
|
|
||||||
Value: value,
|
|
||||||
Tag: tag,
|
|
||||||
}
|
|
||||||
if tag == yaml.NodeTagString && yaml.IsYaml1_1NonString(n) {
|
|
||||||
n.Style = yaml.DoubleQuotedStyle
|
|
||||||
}
|
|
||||||
return func(node *yaml.RNode) error {
|
|
||||||
return node.PipeE(yaml.FieldSetter{
|
|
||||||
Name: key,
|
|
||||||
Value: yaml.NewRNode(n),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package fsslice contains a yaml.Filter to modify a resource if
|
|
||||||
// it matches one or more FieldSpec entries.
|
|
||||||
package fsslice
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fsslice_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
in := &kio.ByteReader{
|
|
||||||
Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`),
|
|
||||||
}
|
|
||||||
fltr := fsslice.Filter{
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
SetValue: filtersutil.SetScalar("green"),
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{Path: "a/b", CreateIfNotPresent: true},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{in},
|
|
||||||
Filters: []kio.Filter{kio.FilterAll(fltr)},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// a:
|
|
||||||
// b: green
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// a:
|
|
||||||
// b: green
|
|
||||||
}
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fsslice
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
var _ yaml.Filter = Filter{}
|
|
||||||
|
|
||||||
// Filter ranges over an FsSlice to modify fields on a single object.
|
|
||||||
// An FsSlice is a range of FieldSpecs. A FieldSpec is a GVK plus a path.
|
|
||||||
type Filter struct {
|
|
||||||
// FieldSpecList list of FieldSpecs to set
|
|
||||||
FsSlice types.FsSlice `yaml:"fsSlice"`
|
|
||||||
|
|
||||||
// SetValue is called on each field that matches one of the FieldSpecs
|
|
||||||
SetValue filtersutil.SetFn
|
|
||||||
|
|
||||||
// CreateKind is used to create fields that do not exist
|
|
||||||
CreateKind yaml.Kind
|
|
||||||
|
|
||||||
// CreateTag is used to set the tag if encountering a null field
|
|
||||||
CreateTag string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (fltr Filter) Filter(obj *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
for i := range fltr.FsSlice {
|
|
||||||
// apply this FieldSpec
|
|
||||||
// create a new filter for each iteration because they
|
|
||||||
// store internal state about the field paths
|
|
||||||
_, err := (&fieldspec.Filter{
|
|
||||||
FieldSpec: fltr.FsSlice[i],
|
|
||||||
SetValue: fltr.SetValue,
|
|
||||||
CreateKind: fltr.CreateKind,
|
|
||||||
CreateTag: fltr.CreateTag,
|
|
||||||
}).Filter(obj)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return obj, nil
|
|
||||||
}
|
|
||||||
@@ -1,121 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package fsslice_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
. "sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type TestCase struct {
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter Filter
|
|
||||||
fsSlice string
|
|
||||||
error string
|
|
||||||
}
|
|
||||||
|
|
||||||
var tests = map[string]TestCase{
|
|
||||||
"empty": {
|
|
||||||
fsSlice: `
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"two": {
|
|
||||||
fsSlice: `
|
|
||||||
- path: a/b
|
|
||||||
group: foo
|
|
||||||
version: v1
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
- path: q/r[]/s/t
|
|
||||||
group: foo
|
|
||||||
version: v1
|
|
||||||
create: true
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
input: `
|
|
||||||
apiVersion: foo/v1
|
|
||||||
kind: Bar
|
|
||||||
q:
|
|
||||||
r:
|
|
||||||
- s: {}
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: foo/v1
|
|
||||||
kind: Bar
|
|
||||||
q:
|
|
||||||
r:
|
|
||||||
- s: {t: e}
|
|
||||||
a:
|
|
||||||
b: e
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
SetValue: filtersutil.SetScalar("e"),
|
|
||||||
CreateKind: yaml.ScalarNode,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
|
||||||
for name := range tests {
|
|
||||||
test := tests[name]
|
|
||||||
t.Run(name, func(t *testing.T) {
|
|
||||||
err := yaml.Unmarshal([]byte(test.fsSlice), &test.filter.FsSlice)
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
|
|
||||||
out := &bytes.Buffer{}
|
|
||||||
rw := &kio.ByteReadWriter{
|
|
||||||
Reader: bytes.NewBufferString(test.input),
|
|
||||||
Writer: out,
|
|
||||||
OmitReaderAnnotations: true,
|
|
||||||
}
|
|
||||||
|
|
||||||
// run the filter
|
|
||||||
err = kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{rw},
|
|
||||||
Filters: []kio.Filter{kio.FilterAll(test.filter)},
|
|
||||||
Outputs: []kio.Writer{rw},
|
|
||||||
}.Execute()
|
|
||||||
if test.error != "" {
|
|
||||||
if !assert.EqualError(t, err, test.error) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
// stop rest of test
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if !assert.NoError(t, err) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
|
|
||||||
// check results
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(test.expected),
|
|
||||||
strings.TrimSpace(out.String())) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package imagetag contains two kio.Filter implementations to cover the
|
|
||||||
// functionality of the kustomize imagetag transformer.
|
|
||||||
//
|
|
||||||
// Filter updates fields based on a FieldSpec and an ImageTag.
|
|
||||||
//
|
|
||||||
// LegacyFilter doesn't use a FieldSpec, and instead only updates image
|
|
||||||
// references if the field is name image and it is underneath a field called
|
|
||||||
// either containers or initContainers.
|
|
||||||
package imagetag
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: FooBar
|
|
||||||
image: nginx
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: BarFoo
|
|
||||||
image: nginx:1.2.1
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
Digest: "12345",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// containers:
|
|
||||||
// - name: FooBar
|
|
||||||
// image: apache@12345
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// containers:
|
|
||||||
// - name: BarFoo
|
|
||||||
// image: apache@12345
|
|
||||||
}
|
|
||||||
|
|
||||||
func ExampleLegacyFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: FooBar
|
|
||||||
image: nginx
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: BarFoo
|
|
||||||
image: nginx:1.2.1
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{LegacyFilter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
Digest: "12345",
|
|
||||||
},
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// containers:
|
|
||||||
// - name: FooBar
|
|
||||||
// image: apache@12345
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// containers:
|
|
||||||
// - name: BarFoo
|
|
||||||
// image: apache@12345
|
|
||||||
}
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter modifies an "image tag", the value used to specify the
|
|
||||||
// name, tag, version digest etc. of (docker) container images
|
|
||||||
// used by a pod template.
|
|
||||||
type Filter struct {
|
|
||||||
// imageTag is the tag we want to apply to the inputs
|
|
||||||
// The name of the image is used as a key, and other fields
|
|
||||||
// can specify a new name, tag, etc.
|
|
||||||
ImageTag types.Image `json:"imageTag,omitempty" yaml:"imageTag,omitempty"`
|
|
||||||
|
|
||||||
// FsSlice contains the FieldSpecs to locate an image field,
|
|
||||||
// e.g. Path: "spec/myContainers[]/image"
|
|
||||||
FsSlice types.FsSlice `json:"fieldSpecs,omitempty" yaml:"fieldSpecs,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(f.filter)).Filter(nodes)
|
|
||||||
return nodes, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) filter(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
// FsSlice is an allowlist, not a denyList, so to deny
|
|
||||||
// something via configuration a new config mechanism is
|
|
||||||
// needed. Until then, hardcode it.
|
|
||||||
if f.isOnDenyList(node) {
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
if err := node.PipeE(fsslice.Filter{
|
|
||||||
FsSlice: f.FsSlice,
|
|
||||||
SetValue: updateImageTagFn(f.ImageTag),
|
|
||||||
}); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) isOnDenyList(node *yaml.RNode) bool {
|
|
||||||
meta, err := node.GetMeta()
|
|
||||||
if err != nil {
|
|
||||||
// A missing 'meta' field will cause problems elsewhere;
|
|
||||||
// ignore it here to keep the signature simple.
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
// Ignore CRDs
|
|
||||||
// https://github.com/kubernetes-sigs/kustomize/issues/890
|
|
||||||
return meta.Kind == `CustomResourceDefinition`
|
|
||||||
}
|
|
||||||
|
|
||||||
func updateImageTagFn(imageTag types.Image) filtersutil.SetFn {
|
|
||||||
return func(node *yaml.RNode) error {
|
|
||||||
return node.PipeE(imageTagUpdater{
|
|
||||||
ImageTag: imageTag,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,674 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestImageTagUpdater_Filter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedOutput string
|
|
||||||
filter Filter
|
|
||||||
fsSlice types.FsSlice
|
|
||||||
}{
|
|
||||||
"ignore CustomResourceDefinition": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apiextensions.k8s.io/v1
|
|
||||||
kind: CustomResourceDefinition
|
|
||||||
metadata:
|
|
||||||
name: whatever
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: apiextensions.k8s.io/v1
|
|
||||||
kind: CustomResourceDefinition
|
|
||||||
metadata:
|
|
||||||
name: whatever
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: whatever
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "whatever",
|
|
||||||
NewName: "theImageShouldNotChangeInACrd",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"legacy multiple images in containers": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: nginx:2.1.2
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache@12345
|
|
||||||
- image: apache@12345
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
Digest: "12345",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"legacy both containers and initContainers": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
initContainers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
initContainers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
NewTag: "3.2.1",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/initContainers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"legacy updates at multiple depths": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
initContainers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
initContainers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
NewTag: "3.2.1",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update with digest": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
image: nginx:1.2.1
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
image: apache@12345
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
Digest: "12345",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"multiple matches in sequence": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: not_nginx@54321
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: not_nginx@54321
|
|
||||||
- image: apache:3.2.1
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
NewTag: "3.2.1",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"new Tag": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:v2
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:v2
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx:v2
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx:v2
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewTag: "v2",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers[]/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"newImage": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: busybox:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: busybox:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: busybox
|
|
||||||
name: nginx-notag
|
|
||||||
- image: busybox@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "busybox",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers[]/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"newImageAndTag": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: busybox:v3
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: busybox:v3
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: busybox:v3
|
|
||||||
name: nginx-notag
|
|
||||||
- image: busybox:v3
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "busybox",
|
|
||||||
NewTag: "v3",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers[]/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"newDigest": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx@sha256:222222222222222222
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx@sha256:222222222222222222
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx@sha256:222222222222222222
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:222222222222222222
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
Digest: "sha256:222222222222222222",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"newImageAndDigest": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.7.9
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: nginx:latest
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx-notag
|
|
||||||
- image: nginx@sha256:111111111111111111
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: busybox@sha256:222222222222222222
|
|
||||||
name: nginx-tagged
|
|
||||||
- image: busybox@sha256:222222222222222222
|
|
||||||
name: nginx-latest
|
|
||||||
- image: foobar:1
|
|
||||||
name: replaced-with-digest
|
|
||||||
- image: postgres:1.8.0
|
|
||||||
name: postgresdb
|
|
||||||
initContainers:
|
|
||||||
- image: busybox@sha256:222222222222222222
|
|
||||||
name: nginx-notag
|
|
||||||
- image: busybox@sha256:222222222222222222
|
|
||||||
name: nginx-sha256
|
|
||||||
- image: alpine:1.8.0
|
|
||||||
name: init-alpine
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "busybox",
|
|
||||||
Digest: "sha256:222222222222222222",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers[]/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"emptyContainers": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
containers: []
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewTag: "v2",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/containers[]/image",
|
|
||||||
// CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"tagWithBraces": {
|
|
||||||
input: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: some.registry.io/my-image:{GENERATED_TAG}
|
|
||||||
name: my-image
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
group: apps
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: deploy1
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: some.registry.io/my-image:my-fixed-tag
|
|
||||||
name: my-image
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "some.registry.io/my-image",
|
|
||||||
NewTag: "my-fixed-tag",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
fsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/containers[]/image",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "spec/template/spec/initContainers[]/image",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
filter := tc.filter
|
|
||||||
filter.FsSlice = tc.fsSlice
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(filtertest.RunFilter(t, tc.input, filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,113 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// LegacyFilter is an implementation of the kio.Filter interface
|
|
||||||
// that scans through the provided kyaml data structure and updates
|
|
||||||
// any values of any image fields that is inside a sequence under
|
|
||||||
// a field called either containers or initContainers. The field is only
|
|
||||||
// update if it has a value that matches and image reference and the name
|
|
||||||
// of the image is a match with the provided ImageTag.
|
|
||||||
type LegacyFilter struct {
|
|
||||||
ImageTag types.Image `json:"imageTag,omitempty" yaml:"imageTag,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = LegacyFilter{}
|
|
||||||
|
|
||||||
func (lf LegacyFilter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(lf.filter)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (lf LegacyFilter) filter(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
meta, err := node.GetMeta()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// We do not make any changes if the type of the resource
|
|
||||||
// is CustomResourceDefinition.
|
|
||||||
if meta.Kind == `CustomResourceDefinition` {
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
fff := findFieldsFilter{
|
|
||||||
fields: []string{"containers", "initContainers"},
|
|
||||||
fieldCallback: checkImageTagsFn(lf.ImageTag),
|
|
||||||
}
|
|
||||||
if err := node.PipeE(fff); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type fieldCallback func(node *yaml.RNode) error
|
|
||||||
|
|
||||||
// findFieldsFilter is an implementation of the kio.Filter
|
|
||||||
// interface. It will walk the data structure and look for fields
|
|
||||||
// that matches the provided list of field names. For each match,
|
|
||||||
// the value of the field will be passed in as a parameter to the
|
|
||||||
// provided fieldCallback.
|
|
||||||
// TODO: move this to kyaml/filterutils
|
|
||||||
type findFieldsFilter struct {
|
|
||||||
fields []string
|
|
||||||
|
|
||||||
fieldCallback fieldCallback
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f findFieldsFilter) Filter(obj *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
return obj, f.walk(obj)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f findFieldsFilter) walk(node *yaml.RNode) error {
|
|
||||||
switch node.YNode().Kind {
|
|
||||||
case yaml.MappingNode:
|
|
||||||
return node.VisitFields(func(n *yaml.MapNode) error {
|
|
||||||
err := f.walk(n.Value)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
key := n.Key.YNode().Value
|
|
||||||
if contains(f.fields, key) {
|
|
||||||
return f.fieldCallback(n.Value)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
case yaml.SequenceNode:
|
|
||||||
return node.VisitElements(func(n *yaml.RNode) error {
|
|
||||||
return f.walk(n)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func contains(slice []string, str string) bool {
|
|
||||||
for _, s := range slice {
|
|
||||||
if s == str {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func checkImageTagsFn(imageTag types.Image) fieldCallback {
|
|
||||||
return func(node *yaml.RNode) error {
|
|
||||||
if node.YNode().Kind != yaml.SequenceNode {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return node.VisitElements(func(n *yaml.RNode) error {
|
|
||||||
// Look up any fields on the provided node that is named
|
|
||||||
// image.
|
|
||||||
return n.PipeE(yaml.Get("image"), imageTagUpdater{
|
|
||||||
ImageTag: imageTag,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,136 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestLegacyImageTag_Filter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedOutput string
|
|
||||||
filter LegacyFilter
|
|
||||||
}{
|
|
||||||
"updates multiple images inside containers": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: nginx:2.1.2
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache@12345
|
|
||||||
- image: apache@12345
|
|
||||||
`,
|
|
||||||
filter: LegacyFilter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
Digest: "12345",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"updates inside both containers and initContainers": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
initContainers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
initContainers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
filter: LegacyFilter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
NewTag: "3.2.1",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"updates on multiple depths": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
initContainers:
|
|
||||||
- image: nginx:1.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: tomcat:1.2.3
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
initContainers:
|
|
||||||
- image: apache:3.2.1
|
|
||||||
- image: apache:1.2.3
|
|
||||||
`,
|
|
||||||
filter: LegacyFilter{
|
|
||||||
ImageTag: types.Image{
|
|
||||||
Name: "nginx",
|
|
||||||
NewName: "apache",
|
|
||||||
NewTag: "3.2.1",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
filter := tc.filter
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(filtertest.RunFilter(t, tc.input, filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package imagetag
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/image"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// imageTagUpdater is an implementation of the kio.Filter interface
|
|
||||||
// that will update the value of the yaml node based on the provided
|
|
||||||
// ImageTag if the current value matches the format of an image reference.
|
|
||||||
type imageTagUpdater struct {
|
|
||||||
Kind string `yaml:"kind,omitempty"`
|
|
||||||
ImageTag types.Image `yaml:"imageTag,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (u imageTagUpdater) Filter(rn *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
if err := yaml.ErrorIfInvalid(rn, yaml.ScalarNode); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
value := rn.YNode().Value
|
|
||||||
|
|
||||||
if !image.IsImageMatched(value, u.ImageTag.Name) {
|
|
||||||
return rn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
name, tag := image.Split(value)
|
|
||||||
if u.ImageTag.NewName != "" {
|
|
||||||
name = u.ImageTag.NewName
|
|
||||||
}
|
|
||||||
if u.ImageTag.NewTag != "" {
|
|
||||||
tag = ":" + u.ImageTag.NewTag
|
|
||||||
}
|
|
||||||
if u.ImageTag.Digest != "" {
|
|
||||||
tag = "@" + u.ImageTag.Digest
|
|
||||||
}
|
|
||||||
|
|
||||||
return rn.Pipe(yaml.FieldSetter{StringValue: name + tag})
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package labels contains a kio.Filter implementation of the kustomize
|
|
||||||
// labels transformer.
|
|
||||||
package labels
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package labels
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
fss := builtinconfig.MakeDefaultConfig().CommonLabels
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{Filter{
|
|
||||||
Labels: map[string]string{
|
|
||||||
"foo": "bar",
|
|
||||||
},
|
|
||||||
FsSlice: fss,
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// labels:
|
|
||||||
// foo: bar
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// labels:
|
|
||||||
// foo: bar
|
|
||||||
}
|
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package labels
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type labelMap map[string]string
|
|
||||||
|
|
||||||
// Filter sets labels.
|
|
||||||
type Filter struct {
|
|
||||||
// Labels is the set of labels to apply to the inputs
|
|
||||||
Labels labelMap `yaml:"labels,omitempty"`
|
|
||||||
|
|
||||||
// FsSlice identifies the label fields.
|
|
||||||
FsSlice types.FsSlice
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
keys := yaml.SortedMapKeys(f.Labels)
|
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(
|
|
||||||
func(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
for _, k := range keys {
|
|
||||||
if err := node.PipeE(fsslice.Filter{
|
|
||||||
FsSlice: f.FsSlice,
|
|
||||||
SetValue: filtersutil.SetEntry(
|
|
||||||
k, f.Labels[k], yaml.NodeTagString),
|
|
||||||
CreateKind: yaml.MappingNode, // Labels are MappingNodes.
|
|
||||||
CreateTag: yaml.NodeTagMap,
|
|
||||||
}); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
})).Filter(nodes)
|
|
||||||
return nodes, err
|
|
||||||
}
|
|
||||||
@@ -1,413 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package labels
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestLabels_Filter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedOutput string
|
|
||||||
filter Filter
|
|
||||||
}{
|
|
||||||
"add": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
auto: ford
|
|
||||||
bean: cannellini
|
|
||||||
clown: emmett kelley
|
|
||||||
dragon: smaug
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"clown": "emmett kelley",
|
|
||||||
"auto": "ford",
|
|
||||||
"dragon": "smaug",
|
|
||||||
"bean": "cannellini",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: superman
|
|
||||||
fiend: luthor
|
|
||||||
bean: cannellini
|
|
||||||
clown: emmett kelley
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"clown": "emmett kelley",
|
|
||||||
"hero": "superman",
|
|
||||||
"fiend": "luthor",
|
|
||||||
"bean": "cannellini",
|
|
||||||
}, FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"data-fieldspecs": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
sleater: kinney
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
sleater: kinney
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
sleater: kinney
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
sleater: kinney
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"sleater": "kinney",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Path: "a/b",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"fieldSpecWithKind": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
cheese: cheddar
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"cheese": "cheddar",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Gvk: resid.Gvk{
|
|
||||||
Kind: "Bar",
|
|
||||||
},
|
|
||||||
Path: "a/b",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"fieldSpecWithVersion": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
cheese: cheddar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"cheese": "cheddar",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Gvk: resid.Gvk{
|
|
||||||
Version: "v1",
|
|
||||||
},
|
|
||||||
Path: "a/b",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"fieldSpecWithVersionInConfigButNoGroupInData": {
|
|
||||||
input: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
cheese: cheddar
|
|
||||||
---
|
|
||||||
apiVersion: v2
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
cheese: cheddar
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"cheese": "cheddar",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Gvk: resid.Gvk{
|
|
||||||
Version: "v1",
|
|
||||||
},
|
|
||||||
Path: "a/b",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"number": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
1: emmett kelley
|
|
||||||
auto: "2"
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"1": "emmett kelley",
|
|
||||||
"auto": "2",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
// test quoting of values which are not considered strings in yaml 1.1
|
|
||||||
"yaml_1_1_compatibility": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
hero: batman
|
|
||||||
fiend: riddler
|
|
||||||
a: "y"
|
|
||||||
b: y1
|
|
||||||
c: "yes"
|
|
||||||
d: yes1
|
|
||||||
e: "true"
|
|
||||||
f: true1
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"a": "y",
|
|
||||||
"b": "y1",
|
|
||||||
"c": "yes",
|
|
||||||
"d": "yes1",
|
|
||||||
"e": "true",
|
|
||||||
"f": "true1",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"null_labels": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels: null
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
labels:
|
|
||||||
a: a1
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Labels: labelMap{
|
|
||||||
"a": "a1",
|
|
||||||
},
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "metadata/labels",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(filtertest_test.RunFilter(t, tc.input, tc.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
// Package nameref contains a kio.Filter implementation of the kustomize
|
|
||||||
// name reference transformer.
|
|
||||||
package nameref
|
|
||||||
@@ -1,402 +0,0 @@
|
|||||||
package nameref
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter updates a name references.
|
|
||||||
type Filter struct {
|
|
||||||
// Referrer refers to another resource X by X's name.
|
|
||||||
// E.g. A Deployment can refer to a ConfigMap.
|
|
||||||
// The Deployment is the Referrer,
|
|
||||||
// the ConfigMap is the ReferralTarget.
|
|
||||||
// This filter seeks to repair the reference in Deployment, given
|
|
||||||
// that the ConfigMap's name may have changed.
|
|
||||||
Referrer *resource.Resource
|
|
||||||
|
|
||||||
// NameFieldToUpdate is the field in the Referrer
|
|
||||||
// that holds the name requiring an update.
|
|
||||||
// This is the field to write.
|
|
||||||
NameFieldToUpdate types.FieldSpec
|
|
||||||
|
|
||||||
// ReferralTarget is the source of the new value for
|
|
||||||
// the name, always in the 'metadata/name' field.
|
|
||||||
// This is the field to read.
|
|
||||||
ReferralTarget resid.Gvk
|
|
||||||
|
|
||||||
// Set of resources to scan to find the ReferralTarget.
|
|
||||||
ReferralCandidates resmap.ResMap
|
|
||||||
}
|
|
||||||
|
|
||||||
// At time of writing, in practice this is called with a slice with only
|
|
||||||
// one entry, the node also referred to be the resource in the Referrer field.
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(f.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
// The node passed in here is the same node as held in Referrer;
|
|
||||||
// that's how the referrer's name field is updated.
|
|
||||||
// Currently, however, this filter still needs the extra methods on Referrer
|
|
||||||
// to consult things like the resource Id, its namespace, etc.
|
|
||||||
// TODO(3455): No filter should use the Resource api; all information
|
|
||||||
// about names should come from annotations, with helper methods
|
|
||||||
// on the RNode object. Resource should get stupider, RNode smarter.
|
|
||||||
func (f Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
if err := f.confirmNodeMatchesReferrer(node); err != nil {
|
|
||||||
// sanity check.
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := node.PipeE(fieldspec.Filter{
|
|
||||||
FieldSpec: f.NameFieldToUpdate,
|
|
||||||
SetValue: f.set,
|
|
||||||
}); err != nil {
|
|
||||||
return nil, errors.Wrapf(
|
|
||||||
err, "updating name reference in '%s' field of '%s'",
|
|
||||||
f.NameFieldToUpdate.Path, f.Referrer.CurId().String())
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function is called on the node found at FieldSpec.Path.
|
|
||||||
// It's some node in the Referrer.
|
|
||||||
func (f Filter) set(node *yaml.RNode) error {
|
|
||||||
if yaml.IsMissingOrNull(node) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
switch node.YNode().Kind {
|
|
||||||
case yaml.ScalarNode:
|
|
||||||
return f.setScalar(node)
|
|
||||||
case yaml.MappingNode:
|
|
||||||
return f.setMapping(node)
|
|
||||||
case yaml.SequenceNode:
|
|
||||||
return applyFilterToSeq(seqFilter{
|
|
||||||
setScalarFn: f.setScalar,
|
|
||||||
setMappingFn: f.setMapping,
|
|
||||||
}, node)
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("node must be a scalar, sequence or map")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// This method used when NameFieldToUpdate doesn't lead to
|
|
||||||
// one scalar field (typically called 'name'), but rather
|
|
||||||
// leads to a map field (called anything). In this case we
|
|
||||||
// must complete the field path, looking for both a 'name'
|
|
||||||
// and a 'namespace' field to help select the proper
|
|
||||||
// ReferralTarget to read the name and namespace from.
|
|
||||||
func (f Filter) setMapping(node *yaml.RNode) error {
|
|
||||||
if node.YNode().Kind != yaml.MappingNode {
|
|
||||||
return fmt.Errorf("expect a mapping node")
|
|
||||||
}
|
|
||||||
nameNode, err := node.Pipe(yaml.FieldMatcher{Name: "name"})
|
|
||||||
if err != nil {
|
|
||||||
return errors.Wrap(err, "trying to match 'name' field")
|
|
||||||
}
|
|
||||||
if nameNode == nil {
|
|
||||||
// This is a _configuration_ error; the field path
|
|
||||||
// specified in NameFieldToUpdate.Path doesn't resolve
|
|
||||||
// to a map with a 'name' field, so we have no idea what
|
|
||||||
// field to update with a new name.
|
|
||||||
return fmt.Errorf("path config error; no 'name' field in node")
|
|
||||||
}
|
|
||||||
candidates, err := f.filterMapCandidatesByNamespace(node)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
oldName := nameNode.YNode().Value
|
|
||||||
referral, err := f.selectReferral(oldName, candidates)
|
|
||||||
if err != nil || referral == nil {
|
|
||||||
// Nil referral means nothing to do.
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
f.recordTheReferral(referral)
|
|
||||||
if referral.GetName() == oldName && referral.GetNamespace() == "" {
|
|
||||||
// The name has not changed, nothing to do.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if err = node.PipeE(yaml.FieldSetter{
|
|
||||||
Name: "name",
|
|
||||||
StringValue: referral.GetName(),
|
|
||||||
}); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if referral.GetNamespace() == "" {
|
|
||||||
// Don't write an empty string into the namespace field, as
|
|
||||||
// it should not replace the value "default". The empty
|
|
||||||
// string is handled as a wild card here, not as an implicit
|
|
||||||
// specification of the "default" k8s namespace.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return node.PipeE(yaml.FieldSetter{
|
|
||||||
Name: "namespace",
|
|
||||||
StringValue: referral.GetNamespace(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) filterMapCandidatesByNamespace(
|
|
||||||
node *yaml.RNode) ([]*resource.Resource, error) {
|
|
||||||
namespaceNode, err := node.Pipe(yaml.FieldMatcher{Name: "namespace"})
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrap(err, "trying to match 'namespace' field")
|
|
||||||
}
|
|
||||||
if namespaceNode == nil {
|
|
||||||
return f.ReferralCandidates.Resources(), nil
|
|
||||||
}
|
|
||||||
namespace := namespaceNode.YNode().Value
|
|
||||||
nsMap := f.ReferralCandidates.GroupedByOriginalNamespace()
|
|
||||||
if candidates, ok := nsMap[namespace]; ok {
|
|
||||||
return candidates, nil
|
|
||||||
}
|
|
||||||
nsMap = f.ReferralCandidates.GroupedByCurrentNamespace()
|
|
||||||
// This could be nil, or an empty list.
|
|
||||||
return nsMap[namespace], nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) setScalar(node *yaml.RNode) error {
|
|
||||||
referral, err := f.selectReferral(
|
|
||||||
node.YNode().Value, f.ReferralCandidates.Resources())
|
|
||||||
if err != nil || referral == nil {
|
|
||||||
// Nil referral means nothing to do.
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
f.recordTheReferral(referral)
|
|
||||||
if referral.GetName() == node.YNode().Value {
|
|
||||||
// The name has not changed, nothing to do.
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return node.PipeE(yaml.FieldSetter{StringValue: referral.GetName()})
|
|
||||||
}
|
|
||||||
|
|
||||||
// In the resource, make a note that it is referred to by the Referrer.
|
|
||||||
func (f Filter) recordTheReferral(referral *resource.Resource) {
|
|
||||||
referral.AppendRefBy(f.Referrer.CurId())
|
|
||||||
}
|
|
||||||
|
|
||||||
// getRoleRefGvk returns a Gvk in the roleRef field. Return error
|
|
||||||
// if the roleRef, roleRef/apiGroup or roleRef/kind is missing.
|
|
||||||
func getRoleRefGvk(res json.Marshaler) (*resid.Gvk, error) {
|
|
||||||
n, err := filtersutil.GetRNode(res)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
roleRef, err := n.Pipe(yaml.Lookup("roleRef"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if roleRef.IsNil() {
|
|
||||||
return nil, fmt.Errorf("roleRef cannot be found in %s", n.MustString())
|
|
||||||
}
|
|
||||||
apiGroup, err := roleRef.Pipe(yaml.Lookup("apiGroup"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if apiGroup.IsNil() {
|
|
||||||
return nil, fmt.Errorf(
|
|
||||||
"apiGroup cannot be found in roleRef %s", roleRef.MustString())
|
|
||||||
}
|
|
||||||
kind, err := roleRef.Pipe(yaml.Lookup("kind"))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if kind.IsNil() {
|
|
||||||
return nil, fmt.Errorf(
|
|
||||||
"kind cannot be found in roleRef %s", roleRef.MustString())
|
|
||||||
}
|
|
||||||
return &resid.Gvk{
|
|
||||||
Group: apiGroup.YNode().Value,
|
|
||||||
Kind: kind.YNode().Value,
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// sieveFunc returns true if the resource argument satisfies some criteria.
|
|
||||||
type sieveFunc func(*resource.Resource) bool
|
|
||||||
|
|
||||||
// doSieve uses a function to accept or ignore resources from a list.
|
|
||||||
// If list is nil, returns immediately.
|
|
||||||
// It's a filter obviously, but that term is overloaded here.
|
|
||||||
func doSieve(list []*resource.Resource, fn sieveFunc) (s []*resource.Resource) {
|
|
||||||
for _, r := range list {
|
|
||||||
if fn(r) {
|
|
||||||
s = append(s, r)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func acceptAll(r *resource.Resource) bool {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func previousNameMatches(name string) sieveFunc {
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
for _, id := range r.PrevIds() {
|
|
||||||
if id.Name == name {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func previousIdSelectedByGvk(gvk *resid.Gvk) sieveFunc {
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
for _, id := range r.PrevIds() {
|
|
||||||
if id.IsSelected(gvk) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the we are updating a 'roleRef/name' field, the 'apiGroup' and 'kind'
|
|
||||||
// fields in the same 'roleRef' map must be considered.
|
|
||||||
// If either object is cluster-scoped (!IsNamespaceableKind), there
|
|
||||||
// can be a referral.
|
|
||||||
// E.g. a RoleBinding (which exists in a namespace) can refer
|
|
||||||
// to a ClusterRole (cluster-scoped) object.
|
|
||||||
// https://kubernetes.io/docs/reference/access-authn-authz/rbac/#role-and-clusterrole
|
|
||||||
// Likewise, a ClusterRole can refer to a Secret (in a namespace).
|
|
||||||
// Objects in different namespaces generally cannot refer to other
|
|
||||||
// with some exceptions (e.g. RoleBinding and ServiceAccount are both
|
|
||||||
// namespaceable, but the former can refer to accounts in other namespaces).
|
|
||||||
func (f Filter) roleRefFilter() sieveFunc {
|
|
||||||
if !strings.HasSuffix(f.NameFieldToUpdate.Path, "roleRef/name") {
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
roleRefGvk, err := getRoleRefGvk(f.Referrer)
|
|
||||||
if err != nil {
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
return previousIdSelectedByGvk(roleRefGvk)
|
|
||||||
}
|
|
||||||
|
|
||||||
func prefixSuffixEquals(other resource.ResCtx) sieveFunc {
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
return r.PrefixesSuffixesEquals(other)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) sameCurrentNamespaceAsReferrer() sieveFunc {
|
|
||||||
referrerCurId := f.Referrer.CurId()
|
|
||||||
if !referrerCurId.IsNamespaceableKind() {
|
|
||||||
// If the referrer is cluster-scoped, let anything through.
|
|
||||||
return acceptAll
|
|
||||||
}
|
|
||||||
return func(r *resource.Resource) bool {
|
|
||||||
if !r.CurId().IsNamespaceableKind() {
|
|
||||||
// Allow cluster-scoped through.
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if r.GetKind() == "ServiceAccount" {
|
|
||||||
// Allow service accounts through, even though they
|
|
||||||
// are in a namespace. A RoleBinding in another namespace
|
|
||||||
// can reference them.
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return referrerCurId.IsNsEquals(r.CurId())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// selectReferral picks the best referral from a list of candidates.
|
|
||||||
func (f Filter) selectReferral(
|
|
||||||
// The name referral that may need to be updated.
|
|
||||||
oldName string,
|
|
||||||
candidates []*resource.Resource) (*resource.Resource, error) {
|
|
||||||
candidates = doSieve(candidates, previousNameMatches(oldName))
|
|
||||||
candidates = doSieve(candidates, previousIdSelectedByGvk(&f.ReferralTarget))
|
|
||||||
candidates = doSieve(candidates, f.roleRefFilter())
|
|
||||||
candidates = doSieve(candidates, f.sameCurrentNamespaceAsReferrer())
|
|
||||||
if len(candidates) == 1 {
|
|
||||||
return candidates[0], nil
|
|
||||||
}
|
|
||||||
candidates = doSieve(candidates, prefixSuffixEquals(f.Referrer))
|
|
||||||
if len(candidates) == 1 {
|
|
||||||
return candidates[0], nil
|
|
||||||
}
|
|
||||||
if len(candidates) == 0 {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
if allNamesAreTheSame(candidates) {
|
|
||||||
// Just take the first one.
|
|
||||||
return candidates[0], nil
|
|
||||||
}
|
|
||||||
ids := getIds(candidates)
|
|
||||||
f.failureDetails(candidates)
|
|
||||||
return nil, fmt.Errorf(" found multiple possible referrals: %s", ids)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) failureDetails(resources []*resource.Resource) {
|
|
||||||
fmt.Printf(
|
|
||||||
"\n**** Too many possible referral targets to referrer:\n%s\n",
|
|
||||||
f.Referrer.MustYaml())
|
|
||||||
for i, r := range resources {
|
|
||||||
fmt.Printf(
|
|
||||||
"--- possible referral %d:\n%s", i, r.MustYaml())
|
|
||||||
fmt.Println("------")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func allNamesAreTheSame(resources []*resource.Resource) bool {
|
|
||||||
name := resources[0].GetName()
|
|
||||||
for i := 1; i < len(resources); i++ {
|
|
||||||
if name != resources[i].GetName() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func getIds(rs []*resource.Resource) string {
|
|
||||||
var result []string
|
|
||||||
for _, r := range rs {
|
|
||||||
result = append(result, r.CurId().String())
|
|
||||||
}
|
|
||||||
return strings.Join(result, ", ")
|
|
||||||
}
|
|
||||||
|
|
||||||
func checkEqual(k, a, b string) error {
|
|
||||||
if a != b {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"node-referrerOriginal '%s' mismatch '%s' != '%s'",
|
|
||||||
k, a, b)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) confirmNodeMatchesReferrer(node *yaml.RNode) error {
|
|
||||||
meta, err := node.GetMeta()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
gvk := f.Referrer.GetGvk()
|
|
||||||
if err = checkEqual(
|
|
||||||
"APIVersion", meta.APIVersion, gvk.ApiVersion()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = checkEqual(
|
|
||||||
"Kind", meta.Kind, gvk.Kind); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = checkEqual(
|
|
||||||
"Name", meta.Name, f.Referrer.GetName()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = checkEqual(
|
|
||||||
"Namespace", meta.Namespace, f.Referrer.GetNamespace()); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -1,789 +0,0 @@
|
|||||||
package nameref
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestNamerefFilter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
referrerOriginal string
|
|
||||||
candidates string
|
|
||||||
referrerFinal string
|
|
||||||
filter Filter
|
|
||||||
originalNames []string
|
|
||||||
}{
|
|
||||||
"simple scalar": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: NotSecret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "newName2"},
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: newName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sequence": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
seq:
|
|
||||||
- oldName1
|
|
||||||
- oldName2
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: NotSecret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName1", "newName2"},
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
seq:
|
|
||||||
- newName
|
|
||||||
- oldName2
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "seq"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"mapping": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
map:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: NotSecret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "newName2"},
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
map:
|
|
||||||
name: newName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"mapping with namespace": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
namespace: someNs
|
|
||||||
map:
|
|
||||||
name: oldName
|
|
||||||
namespace: someNs
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
namespace: someNs
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: NotSecret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: thirdName
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName", "oldName"},
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
namespace: someNs
|
|
||||||
map:
|
|
||||||
name: newName
|
|
||||||
namespace: someNs
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"null value": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
map:
|
|
||||||
name: null
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: NotSecret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "newName2"},
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
map:
|
|
||||||
name: null
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "map"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
tc.filter.Referrer = referrer
|
|
||||||
|
|
||||||
resMapFactory := resmap.NewFactory(factory, nil)
|
|
||||||
candidatesRes, err := factory.SliceFromBytesWithNames(
|
|
||||||
tc.originalNames, []byte(tc.candidates))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
|
||||||
tc.filter.ReferralCandidates = candidates
|
|
||||||
|
|
||||||
result := filtertest_test.RunFilter(t, tc.referrerOriginal, tc.filter)
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.referrerFinal),
|
|
||||||
strings.TrimSpace(result)) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestNamerefFilterUnhappy(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
referrerOriginal string
|
|
||||||
candidates string
|
|
||||||
referrerFinal string
|
|
||||||
filter Filter
|
|
||||||
originalNames []string
|
|
||||||
}{
|
|
||||||
"multiple match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
referrerFinal: "",
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"no name": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
notName: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
referrerFinal: "",
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
tc.filter.Referrer = referrer
|
|
||||||
|
|
||||||
resMapFactory := resmap.NewFactory(factory, nil)
|
|
||||||
candidatesRes, err := factory.SliceFromBytesWithNames(
|
|
||||||
tc.originalNames, []byte(tc.candidates))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
|
||||||
tc.filter.ReferralCandidates = candidates
|
|
||||||
|
|
||||||
_, err = filtertest_test.RunFilterE(t, tc.referrerOriginal, tc.filter)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatalf("expect an error")
|
|
||||||
}
|
|
||||||
if tc.referrerFinal != "" && !assert.EqualError(t, err, tc.referrerFinal) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestCandidatesWithDifferentPrefixSuffix(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
referrerOriginal string
|
|
||||||
candidates string
|
|
||||||
referrerFinal string
|
|
||||||
filter Filter
|
|
||||||
originalNames []string
|
|
||||||
prefix []string
|
|
||||||
suffix []string
|
|
||||||
inputPrefix string
|
|
||||||
inputSuffix string
|
|
||||||
err bool
|
|
||||||
}{
|
|
||||||
"prefix match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix1", "prefix2"},
|
|
||||||
suffix: []string{"", "suffix2"},
|
|
||||||
inputPrefix: "prefix1",
|
|
||||||
inputSuffix: "",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: newName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
"suffix match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"", "prefix2"},
|
|
||||||
suffix: []string{"suffix1", "suffix2"},
|
|
||||||
inputPrefix: "",
|
|
||||||
inputSuffix: "suffix1",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: newName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
"prefix suffix both match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix1", "prefix2"},
|
|
||||||
suffix: []string{"suffix1", "suffix2"},
|
|
||||||
inputPrefix: "prefix1",
|
|
||||||
inputSuffix: "suffix1",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: newName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
"multiple match: both": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix", "prefix"},
|
|
||||||
suffix: []string{"suffix", "suffix"},
|
|
||||||
inputPrefix: "prefix",
|
|
||||||
inputSuffix: "suffix",
|
|
||||||
referrerFinal: "",
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: true,
|
|
||||||
},
|
|
||||||
"multiple match: only prefix": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix", "prefix"},
|
|
||||||
suffix: []string{"", ""},
|
|
||||||
inputPrefix: "prefix",
|
|
||||||
inputSuffix: "",
|
|
||||||
referrerFinal: "",
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: true,
|
|
||||||
},
|
|
||||||
"multiple match: only suffix": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"", ""},
|
|
||||||
suffix: []string{"suffix", "suffix"},
|
|
||||||
inputPrefix: "",
|
|
||||||
inputSuffix: "suffix",
|
|
||||||
referrerFinal: "",
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: true,
|
|
||||||
},
|
|
||||||
"no match: neither match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix1", "prefix2"},
|
|
||||||
suffix: []string{"suffix1", "suffix2"},
|
|
||||||
inputPrefix: "prefix",
|
|
||||||
inputSuffix: "suffix",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
"no match: prefix doesn't match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix1", "prefix2"},
|
|
||||||
suffix: []string{"suffix", "suffix"},
|
|
||||||
inputPrefix: "prefix",
|
|
||||||
inputSuffix: "suffix",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
"no match: suffix doesn't match": {
|
|
||||||
referrerOriginal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
candidates: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName
|
|
||||||
---
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Secret
|
|
||||||
metadata:
|
|
||||||
name: newName2
|
|
||||||
`,
|
|
||||||
originalNames: []string{"oldName", "oldName"},
|
|
||||||
prefix: []string{"prefix", "prefix"},
|
|
||||||
suffix: []string{"suffix1", "suffix2"},
|
|
||||||
inputPrefix: "prefix",
|
|
||||||
inputSuffix: "suffix",
|
|
||||||
referrerFinal: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
ref:
|
|
||||||
name: oldName
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
NameFieldToUpdate: types.FieldSpec{Path: "ref/name"},
|
|
||||||
ReferralTarget: resid.Gvk{
|
|
||||||
Group: "apps",
|
|
||||||
Version: "v1",
|
|
||||||
Kind: "Secret",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
err: false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
factory := provider.NewDefaultDepProvider().GetResourceFactory()
|
|
||||||
referrer, err := factory.FromBytes([]byte(tc.referrerOriginal))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if tc.inputPrefix != "" {
|
|
||||||
referrer.AddNamePrefix(tc.inputPrefix)
|
|
||||||
}
|
|
||||||
if tc.inputSuffix != "" {
|
|
||||||
referrer.AddNameSuffix(tc.inputSuffix)
|
|
||||||
}
|
|
||||||
tc.filter.Referrer = referrer
|
|
||||||
|
|
||||||
resMapFactory := resmap.NewFactory(factory, nil)
|
|
||||||
candidatesRes, err := factory.SliceFromBytesWithNames(
|
|
||||||
tc.originalNames, []byte(tc.candidates))
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
for i := range candidatesRes {
|
|
||||||
if tc.prefix[i] != "" {
|
|
||||||
candidatesRes[i].AddNamePrefix(tc.prefix[i])
|
|
||||||
}
|
|
||||||
if tc.suffix[i] != "" {
|
|
||||||
candidatesRes[i].AddNameSuffix(tc.suffix[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
candidates := resMapFactory.FromResourceSlice(candidatesRes)
|
|
||||||
tc.filter.ReferralCandidates = candidates
|
|
||||||
|
|
||||||
if !tc.err {
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.referrerFinal),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest_test.RunFilter(
|
|
||||||
t, tc.referrerOriginal, tc.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
_, err := filtertest_test.RunFilterE(
|
|
||||||
t, tc.referrerOriginal, tc.filter)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatalf("an error is expected")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
package nameref
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type setFn func(*yaml.RNode) error
|
|
||||||
|
|
||||||
type seqFilter struct {
|
|
||||||
setScalarFn setFn
|
|
||||||
setMappingFn setFn
|
|
||||||
}
|
|
||||||
|
|
||||||
func (sf seqFilter) Filter(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
if yaml.IsMissingOrNull(node) {
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
switch node.YNode().Kind {
|
|
||||||
case yaml.ScalarNode:
|
|
||||||
// Kind: Role/ClusterRole
|
|
||||||
// FieldSpec is rules.resourceNames
|
|
||||||
err := sf.setScalarFn(node)
|
|
||||||
return node, err
|
|
||||||
case yaml.MappingNode:
|
|
||||||
// Kind: RoleBinding/ClusterRoleBinding
|
|
||||||
// FieldSpec is subjects
|
|
||||||
// Note: The corresponding fieldSpec had been changed from
|
|
||||||
// from path: subjects/name to just path: subjects. This is
|
|
||||||
// what get mutatefield to request the mapping of the whole
|
|
||||||
// map containing namespace and name instead of just a simple
|
|
||||||
// string field containing the name
|
|
||||||
err := sf.setMappingFn(node)
|
|
||||||
return node, err
|
|
||||||
default:
|
|
||||||
return node, fmt.Errorf(
|
|
||||||
"%#v is expected to be either a string or a map of string", node)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// applyFilterToSeq will apply the filter to each element in the sequence node
|
|
||||||
func applyFilterToSeq(filter yaml.Filter, node *yaml.RNode) error {
|
|
||||||
if node.YNode().Kind != yaml.SequenceNode {
|
|
||||||
return fmt.Errorf("expect a sequence node but got %v", node.YNode().Kind)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, elem := range node.Content() {
|
|
||||||
rnode := yaml.NewRNode(elem)
|
|
||||||
err := rnode.PipeE(filter)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
package nameref
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func SeqFilter(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
if node.YNode().Value == "aaa" {
|
|
||||||
node.YNode().SetString("ccc")
|
|
||||||
}
|
|
||||||
return node, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestApplyFilterToSeq(t *testing.T) {
|
|
||||||
fltr := yaml.FilterFunc(SeqFilter)
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expect string
|
|
||||||
}{
|
|
||||||
"replace in seq": {
|
|
||||||
input: `
|
|
||||||
- aaa
|
|
||||||
- bbb`,
|
|
||||||
expect: `
|
|
||||||
- ccc
|
|
||||||
- bbb`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
node, err := yaml.Parse(tc.input)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
err = applyFilterToSeq(fltr, node)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expect),
|
|
||||||
strings.TrimSpace(node.MustString())) {
|
|
||||||
t.Fatalf("expect:\n%s\nactual:\n%s",
|
|
||||||
strings.TrimSpace(tc.expect),
|
|
||||||
strings.TrimSpace(node.MustString()))
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestApplyFilterToSeqUnhappy(t *testing.T) {
|
|
||||||
fltr := yaml.FilterFunc(SeqFilter)
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
}{
|
|
||||||
"replace in seq": {
|
|
||||||
input: `
|
|
||||||
aaa`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
node, err := yaml.Parse(tc.input)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
err = applyFilterToSeq(fltr, node)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatalf("expect an error")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package namespace contains a kio.Filter implementation of the kustomize
|
|
||||||
// namespace transformer.
|
|
||||||
//
|
|
||||||
// Special cases for known Kubernetes resources have been hardcoded in addition
|
|
||||||
// to those defined by the FsSlice.
|
|
||||||
package namespace
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package namespace_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/namespace"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
fss := builtinconfig.MakeDefaultConfig().NameSpace
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: bar
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{namespace.Filter{Namespace: "app", FsSlice: fss}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// namespace: app
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// namespace: app
|
|
||||||
}
|
|
||||||
@@ -1,169 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package namespace
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fsslice"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Filter struct {
|
|
||||||
// Namespace is the namespace to apply to the inputs
|
|
||||||
Namespace string `yaml:"namespace,omitempty"`
|
|
||||||
|
|
||||||
// FsSlice contains the FieldSpecs to locate the namespace field
|
|
||||||
FsSlice types.FsSlice `json:"fieldSpecs,omitempty" yaml:"fieldSpecs,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (ns Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(ns.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run runs the filter on a single node rather than a slice
|
|
||||||
func (ns Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
// hacks for hardcoded types -- :(
|
|
||||||
if err := ns.hacks(node); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the fieldspecs that are for hardcoded fields. The fieldspecs
|
|
||||||
// exist for backwards compatibility with other implementations
|
|
||||||
// of this transformation.
|
|
||||||
// This implementation of the namespace transformation
|
|
||||||
// Does not use the fieldspecs for implementing cases which
|
|
||||||
// require hardcoded logic.
|
|
||||||
ns.FsSlice = ns.removeFieldSpecsForHacks(ns.FsSlice)
|
|
||||||
|
|
||||||
// transformations based on data -- :)
|
|
||||||
err := node.PipeE(fsslice.Filter{
|
|
||||||
FsSlice: ns.FsSlice,
|
|
||||||
SetValue: filtersutil.SetScalar(ns.Namespace),
|
|
||||||
CreateKind: yaml.ScalarNode, // Namespace is a ScalarNode
|
|
||||||
CreateTag: yaml.NodeTagString,
|
|
||||||
})
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// hacks applies the namespace transforms that are hardcoded rather
|
|
||||||
// than specified through FieldSpecs.
|
|
||||||
func (ns Filter) hacks(obj *yaml.RNode) error {
|
|
||||||
meta, err := obj.GetMeta()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := ns.metaNamespaceHack(obj, meta); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return ns.roleBindingHack(obj, meta)
|
|
||||||
}
|
|
||||||
|
|
||||||
// metaNamespaceHack is a hack for implementing the namespace transform
|
|
||||||
// for the metadata.namespace field on namespace scoped resources.
|
|
||||||
// namespace scoped resources are determined by NOT being present
|
|
||||||
// in a hard-coded list of cluster-scoped resource types (by apiVersion and kind).
|
|
||||||
//
|
|
||||||
// This hack should be updated to allow individual resources to specify
|
|
||||||
// if they are cluster scoped through either an annotation on the resources,
|
|
||||||
// or through inlined OpenAPI on the resource as a YAML comment.
|
|
||||||
func (ns Filter) metaNamespaceHack(obj *yaml.RNode, meta yaml.ResourceMeta) error {
|
|
||||||
gvk := fieldspec.GetGVK(meta)
|
|
||||||
if !gvk.IsNamespaceableKind() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
f := fsslice.Filter{
|
|
||||||
FsSlice: []types.FieldSpec{
|
|
||||||
{Path: types.MetadataNamespacePath, CreateIfNotPresent: true},
|
|
||||||
},
|
|
||||||
SetValue: filtersutil.SetScalar(ns.Namespace),
|
|
||||||
CreateKind: yaml.ScalarNode, // Namespace is a ScalarNode
|
|
||||||
}
|
|
||||||
_, err := f.Filter(obj)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// roleBindingHack is a hack for implementing the namespace transform
|
|
||||||
// for RoleBinding and ClusterRoleBinding resource types.
|
|
||||||
// RoleBinding and ClusterRoleBinding have namespace set on
|
|
||||||
// elements of the "subjects" field if and only if the subject elements
|
|
||||||
// "name" is "default". Otherwise the namespace is not set.
|
|
||||||
//
|
|
||||||
// Example:
|
|
||||||
//
|
|
||||||
// kind: RoleBinding
|
|
||||||
// subjects:
|
|
||||||
// - name: "default" # this will have the namespace set
|
|
||||||
// ...
|
|
||||||
// - name: "something-else" # this will not have the namespace set
|
|
||||||
// ...
|
|
||||||
func (ns Filter) roleBindingHack(obj *yaml.RNode, meta yaml.ResourceMeta) error {
|
|
||||||
if meta.Kind != roleBindingKind && meta.Kind != clusterRoleBindingKind {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Lookup the namespace field on all elements.
|
|
||||||
// We should change the fieldspec so this isn't necessary.
|
|
||||||
obj, err := obj.Pipe(yaml.Lookup(subjectsField))
|
|
||||||
if err != nil || yaml.IsMissingOrNull(obj) {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// add the namespace to each "subject" with name: default
|
|
||||||
err = obj.VisitElements(func(o *yaml.RNode) error {
|
|
||||||
// copied from kunstruct based kustomize NamespaceTransformer plugin
|
|
||||||
// The only case we need to force the namespace
|
|
||||||
// if for the "service account". "default" is
|
|
||||||
// kind of hardcoded here for right now.
|
|
||||||
name, err := o.Pipe(
|
|
||||||
yaml.Lookup("name"), yaml.Match("default"),
|
|
||||||
)
|
|
||||||
if err != nil || yaml.IsMissingOrNull(name) {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the namespace for the default account
|
|
||||||
v := yaml.NewScalarRNode(ns.Namespace)
|
|
||||||
return o.PipeE(
|
|
||||||
yaml.LookupCreate(yaml.ScalarNode, "namespace"),
|
|
||||||
yaml.FieldSetter{Value: v},
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// removeFieldSpecsForHacks removes from the list fieldspecs that
|
|
||||||
// have hardcoded implementations
|
|
||||||
func (ns Filter) removeFieldSpecsForHacks(fs types.FsSlice) types.FsSlice {
|
|
||||||
var val types.FsSlice
|
|
||||||
for i := range fs {
|
|
||||||
// implemented by metaNamespaceHack
|
|
||||||
if fs[i].Path == types.MetadataNamespacePath {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// implemented by roleBindingHack
|
|
||||||
if fs[i].Kind == roleBindingKind && fs[i].Path == subjectsField {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// implemented by roleBindingHack
|
|
||||||
if fs[i].Kind == clusterRoleBindingKind && fs[i].Path == subjectsField {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
val = append(val, fs[i])
|
|
||||||
}
|
|
||||||
return val
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
subjectsField = "subjects"
|
|
||||||
roleBindingKind = "RoleBinding"
|
|
||||||
clusterRoleBindingKind = "ClusterRoleBinding"
|
|
||||||
)
|
|
||||||
@@ -1,311 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package namespace_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/namespace"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var tests = []TestCase{
|
|
||||||
{
|
|
||||||
name: "add",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "foo"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "null_ns",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: null
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: null
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "foo"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "add-recurse",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "foo"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "update",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
# update this namespace
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: bar
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
# update this namespace
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "foo"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "update-rolebinding",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: bar
|
|
||||||
metadata:
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: bar
|
|
||||||
metadata:
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
metadata:
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: RoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
namespace: foo
|
|
||||||
metadata:
|
|
||||||
namespace: bar
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "bar"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "update-clusterrolebinding",
|
|
||||||
input: `
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: foo
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: default
|
|
||||||
namespace: bar
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
---
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
subjects:
|
|
||||||
- name: something
|
|
||||||
namespace: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "bar"},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
name: "data-fieldspecs",
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: foo
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: foo
|
|
||||||
`,
|
|
||||||
filter: namespace.Filter{Namespace: "foo"},
|
|
||||||
fsslice: []types.FieldSpec{
|
|
||||||
{
|
|
||||||
Path: "a/b/c",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
type TestCase struct {
|
|
||||||
name string
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter namespace.Filter
|
|
||||||
fsslice types.FsSlice
|
|
||||||
}
|
|
||||||
|
|
||||||
var config = builtinconfig.MakeDefaultConfig()
|
|
||||||
|
|
||||||
func TestNamespace_Filter(t *testing.T) {
|
|
||||||
for i := range tests {
|
|
||||||
test := tests[i]
|
|
||||||
t.Run(test.name, func(t *testing.T) {
|
|
||||||
test.filter.FsSlice = append(config.NameSpace, test.fsslice...)
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(test.expected),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest_test.RunFilter(t, test.input, test.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package namespace contains a kio.Filter implementation of the kustomize
|
|
||||||
// patchjson6902 transformer
|
|
||||||
package patchjson6902
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patchjson6902
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
namespace: bar
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{
|
|
||||||
Filter{
|
|
||||||
Patch: `
|
|
||||||
- op: replace
|
|
||||||
path: /metadata/namespace
|
|
||||||
value: "ns"
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// namespace: ns
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// namespace: ns
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patchjson6902
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
jsonpatch "github.com/evanphx/json-patch"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
k8syaml "sigs.k8s.io/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Filter struct {
|
|
||||||
Patch string
|
|
||||||
|
|
||||||
decodedPatch jsonpatch.Patch
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (pf Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
decodedPatch, err := pf.decodePatch()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
pf.decodedPatch = decodedPatch
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(pf.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (pf Filter) decodePatch() (jsonpatch.Patch, error) {
|
|
||||||
patch := pf.Patch
|
|
||||||
// If the patch doesn't look like a JSON6902 patch, we
|
|
||||||
// try to parse it to json.
|
|
||||||
if !strings.HasPrefix(pf.Patch, "[") {
|
|
||||||
p, err := k8syaml.YAMLToJSON([]byte(patch))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
patch = string(p)
|
|
||||||
}
|
|
||||||
decodedPatch, err := jsonpatch.DecodePatch([]byte(patch))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return decodedPatch, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (pf Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
// We don't actually use the kyaml library for manipulating the
|
|
||||||
// yaml here. We just marshal it to json and rely on the
|
|
||||||
// jsonpatch library to take care of applying the patch.
|
|
||||||
// This means ordering might not be preserved with this filter.
|
|
||||||
b, err := node.MarshalJSON()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
res, err := pf.decodedPatch.Apply(b)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
err = node.UnmarshalJSON(res)
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patchjson6902
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
)
|
|
||||||
|
|
||||||
const input = `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
replica: 2
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
old-label: old-value
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`
|
|
||||||
|
|
||||||
func TestSomething(t *testing.T) {
|
|
||||||
testCases := []struct {
|
|
||||||
testName string
|
|
||||||
input string
|
|
||||||
filter Filter
|
|
||||||
expectedOutput string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
testName: "single operation, json",
|
|
||||||
input: input,
|
|
||||||
filter: Filter{
|
|
||||||
Patch: `[
|
|
||||||
{"op": "replace", "path": "/spec/replica", "value": 5}
|
|
||||||
]`,
|
|
||||||
},
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
replica: 5
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
old-label: old-value
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "multiple operations, json",
|
|
||||||
input: input,
|
|
||||||
filter: Filter{
|
|
||||||
Patch: `[
|
|
||||||
{"op": "replace", "path": "/spec/template/spec/containers/0/name", "value": "my-nginx"},
|
|
||||||
{"op": "add", "path": "/spec/replica", "value": 999},
|
|
||||||
{"op": "add", "path": "/spec/template/spec/containers/0/command", "value": ["arg1", "arg2", "arg3"]}
|
|
||||||
]`,
|
|
||||||
},
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
replica: 999
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
old-label: old-value
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- command:
|
|
||||||
- arg1
|
|
||||||
- arg2
|
|
||||||
- arg3
|
|
||||||
image: nginx
|
|
||||||
name: my-nginx
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "single operation, yaml",
|
|
||||||
input: input,
|
|
||||||
filter: Filter{
|
|
||||||
Patch: `
|
|
||||||
- op: replace
|
|
||||||
path: /spec/replica
|
|
||||||
value: 5
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
replica: 5
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
old-label: old-value
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
name: nginx
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "multiple operations, yaml",
|
|
||||||
input: input,
|
|
||||||
filter: Filter{
|
|
||||||
Patch: `
|
|
||||||
- op: replace
|
|
||||||
path: /spec/template/spec/containers/0/name
|
|
||||||
value: my-nginx
|
|
||||||
- op: add
|
|
||||||
path: /spec/replica
|
|
||||||
value: 999
|
|
||||||
- op: add
|
|
||||||
path: /spec/template/spec/containers/0/command
|
|
||||||
value:
|
|
||||||
- arg1
|
|
||||||
- arg2
|
|
||||||
- arg3
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
expectedOutput: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
replica: 999
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
old-label: old-value
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- command:
|
|
||||||
- arg1
|
|
||||||
- arg2
|
|
||||||
- arg3
|
|
||||||
image: nginx
|
|
||||||
name: my-nginx
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tc := range testCases {
|
|
||||||
t.Run(tc.testName, func(t *testing.T) {
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest.RunFilter(t, tc.input, tc.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package patchstrategicmerge contains a kio.Filter implementation of the
|
|
||||||
// kustomize strategic merge patch transformer.
|
|
||||||
package patchstrategicmerge
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patchstrategicmerge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
replicas: 3
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{Filter{
|
|
||||||
Patch: yaml.MustParse(`
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
containers:
|
|
||||||
- image: nginx
|
|
||||||
`),
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// replicas: 3
|
|
||||||
// template:
|
|
||||||
// containers:
|
|
||||||
// - image: nginx
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patchstrategicmerge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml/merge2"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Filter struct {
|
|
||||||
Patch *yaml.RNode
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
// Filter does a strategic merge patch, which can delete nodes.
|
|
||||||
func (pf Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
var result []*yaml.RNode
|
|
||||||
for i := range nodes {
|
|
||||||
r, err := merge2.Merge(
|
|
||||||
pf.Patch, nodes[i],
|
|
||||||
yaml.MergeOptions{
|
|
||||||
ListIncreaseDirection: yaml.MergeOptionsListPrepend,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if !konfig.FlagEnableKyamlDefaultValue || r != nil {
|
|
||||||
result = append(result, r)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result, nil
|
|
||||||
}
|
|
||||||
@@ -1,749 +0,0 @@
|
|||||||
package patchstrategicmerge
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
patch *yaml.RNode
|
|
||||||
expected string
|
|
||||||
}{
|
|
||||||
"simple": {
|
|
||||||
input: `apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 1
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 999
|
|
||||||
`),
|
|
||||||
expected: `apiVersion: v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: clown
|
|
||||||
spec:
|
|
||||||
numReplicas: 999
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"nullMapEntry1": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
B:
|
|
||||||
C: Z
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"nullMapEntry2": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
B:
|
|
||||||
C: Z
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: my-foo
|
|
||||||
spec:
|
|
||||||
bar:
|
|
||||||
C: Z
|
|
||||||
D: W
|
|
||||||
baz:
|
|
||||||
hello: world
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"simple patch": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
metadata:
|
|
||||||
name: yourDeploy
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: yourDeploy
|
|
||||||
kind: Deployment
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"container patch": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: foo1
|
|
||||||
- name: foo2
|
|
||||||
- name: foo3
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: foo0
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: foo0
|
|
||||||
- name: foo1
|
|
||||||
- name: foo2
|
|
||||||
- name: foo3
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"volumes patch": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
volumes:
|
|
||||||
- name: foo1
|
|
||||||
- name: foo2
|
|
||||||
- name: foo3
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
volumes:
|
|
||||||
- name: foo0
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
volumes:
|
|
||||||
- name: foo0
|
|
||||||
- name: foo1
|
|
||||||
- name: foo2
|
|
||||||
- name: foo3
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"nested patch": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
args:
|
|
||||||
- abc
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
args:
|
|
||||||
- def
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
kind: Deployment
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: nginx
|
|
||||||
args:
|
|
||||||
- def
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"remove mapping - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
$patch: delete
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers: []
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"replace mapping - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
$patch: replace
|
|
||||||
containers:
|
|
||||||
- name: new
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: new
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"merge mapping - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test1
|
|
||||||
$patch: merge
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test1
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"remove list - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- whatever
|
|
||||||
- $patch: delete
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec: {}
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"replace list - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: replace
|
|
||||||
image: replace
|
|
||||||
- $patch: replace
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: replace
|
|
||||||
image: replace
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"merge list - directive": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test2
|
|
||||||
image: test2
|
|
||||||
- $patch: merge
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: myDeploy
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: test2
|
|
||||||
image: test2
|
|
||||||
- name: test
|
|
||||||
image: test
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"list map keys - add a port, no names": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
- containerPort: 80
|
|
||||||
protocol: UDP
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
- containerPort: 80
|
|
||||||
protocol: UDP
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"list map keys - add name to port": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
name: UDP-name-patch
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
name: UDP-name-patch
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"list map keys - replace port name": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
name: UDP-name-original
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
name: TCP-name-original
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
name: UDP-name-patch
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: UDP
|
|
||||||
name: UDP-name-patch
|
|
||||||
- containerPort: 8080
|
|
||||||
protocol: TCP
|
|
||||||
name: TCP-name-original
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
"list map keys - add a port, no protocol": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 8080
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 80
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- image: test-image
|
|
||||||
name: test-deployment
|
|
||||||
ports:
|
|
||||||
- containerPort: 80
|
|
||||||
- containerPort: 8080
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Test for issue #3513
|
|
||||||
// Currently broken; when one port has only containerPort, the output
|
|
||||||
// should not merge containerPort 8301 together
|
|
||||||
// This occurs because when protocol is missing on the first port,
|
|
||||||
// the merge code uses [containerPort] as the merge key rather than
|
|
||||||
// [containerPort, protocol]
|
|
||||||
"list map keys - protocol only present on some ports": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: consul
|
|
||||||
image: "hashicorp/consul:1.9.1"
|
|
||||||
ports:
|
|
||||||
- containerPort: 8500
|
|
||||||
name: http
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "TCP"
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "UDP"
|
|
||||||
`,
|
|
||||||
patch: yaml.MustParse(`
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
labels:
|
|
||||||
test: label
|
|
||||||
`),
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: test-deployment
|
|
||||||
labels:
|
|
||||||
test: label
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
spec:
|
|
||||||
containers:
|
|
||||||
- name: consul
|
|
||||||
image: "hashicorp/consul:1.9.1"
|
|
||||||
ports:
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "TCP"
|
|
||||||
- containerPort: 8301
|
|
||||||
protocol: "UDP"
|
|
||||||
- containerPort: 8500
|
|
||||||
name: http
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
f := Filter{
|
|
||||||
Patch: tc.patch,
|
|
||||||
}
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expected),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest.RunFilter(t, tc.input, f))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package prefixsuffix contains a kio.Filter implementation of the kustomize
|
|
||||||
// PrefixSuffixTransformer.
|
|
||||||
package prefixsuffix
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package prefixsuffix_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/prefixsuffix"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{prefixsuffix.Filter{
|
|
||||||
Prefix: "baz-", FieldSpec: types.FieldSpec{Path: "metadata/name"}}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: baz-instance
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: baz-instance
|
|
||||||
}
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package prefixsuffix
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter applies resource name prefix's and suffix's using the fieldSpecs
|
|
||||||
type Filter struct {
|
|
||||||
Prefix string `json:"prefix,omitempty" yaml:"prefix,omitempty"`
|
|
||||||
Suffix string `json:"suffix,omitempty" yaml:"suffix,omitempty"`
|
|
||||||
|
|
||||||
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(f.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
err := node.PipeE(fieldspec.Filter{
|
|
||||||
FieldSpec: f.FieldSpec,
|
|
||||||
SetValue: f.evaluateField,
|
|
||||||
CreateKind: yaml.ScalarNode, // Name is a ScalarNode
|
|
||||||
CreateTag: yaml.NodeTagString,
|
|
||||||
})
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) evaluateField(node *yaml.RNode) error {
|
|
||||||
return filtersutil.SetScalar(fmt.Sprintf(
|
|
||||||
"%s%s%s", f.Prefix, node.YNode().Value, f.Suffix))(node)
|
|
||||||
}
|
|
||||||
@@ -1,158 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package prefixsuffix_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/prefixsuffix"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var tests = map[string]TestCase{
|
|
||||||
"prefix": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: foo-instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: foo-instance
|
|
||||||
`,
|
|
||||||
filter: prefixsuffix.Filter{Prefix: "foo-"},
|
|
||||||
fs: types.FieldSpec{Path: "metadata/name"},
|
|
||||||
},
|
|
||||||
|
|
||||||
"suffix": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance-foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance-foo
|
|
||||||
`,
|
|
||||||
filter: prefixsuffix.Filter{Suffix: "-foo"},
|
|
||||||
fs: types.FieldSpec{Path: "metadata/name"},
|
|
||||||
},
|
|
||||||
|
|
||||||
"prefix-suffix": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: bar-instance-foo
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: bar-instance-foo
|
|
||||||
`,
|
|
||||||
filter: prefixsuffix.Filter{Prefix: "bar-", Suffix: "-foo"},
|
|
||||||
fs: types.FieldSpec{Path: "metadata/name"},
|
|
||||||
},
|
|
||||||
|
|
||||||
"data-fieldspecs": {
|
|
||||||
input: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: d
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: d
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: foo-d
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
a:
|
|
||||||
b:
|
|
||||||
c: foo-d
|
|
||||||
`,
|
|
||||||
filter: prefixsuffix.Filter{Prefix: "foo-"},
|
|
||||||
fs: types.FieldSpec{Path: "a/b/c"},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
type TestCase struct {
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter prefixsuffix.Filter
|
|
||||||
fs types.FieldSpec
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
|
||||||
for name := range tests {
|
|
||||||
test := tests[name]
|
|
||||||
t.Run(name, func(t *testing.T) {
|
|
||||||
test.filter.FieldSpec = test.fs
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(test.expected),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest_test.RunFilter(t, test.input, test.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
// Package refvar contains a kio.Filter implementation of the kustomize
|
|
||||||
// refvar transformer (find and replace $(FOO) style variables in strings).
|
|
||||||
package refvar
|
|
||||||
@@ -1,147 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package refvar
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"log"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
operator = '$'
|
|
||||||
referenceOpener = '('
|
|
||||||
referenceCloser = ')'
|
|
||||||
)
|
|
||||||
|
|
||||||
// syntaxWrap returns the input string wrapped by the expansion syntax.
|
|
||||||
func syntaxWrap(input string) string {
|
|
||||||
var sb strings.Builder
|
|
||||||
sb.WriteByte(operator)
|
|
||||||
sb.WriteByte(referenceOpener)
|
|
||||||
sb.WriteString(input)
|
|
||||||
sb.WriteByte(referenceCloser)
|
|
||||||
return sb.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
// MappingFunc maps a string to anything.
|
|
||||||
type MappingFunc func(string) interface{}
|
|
||||||
|
|
||||||
// MakePrimitiveReplacer returns a MappingFunc that uses a map to do
|
|
||||||
// replacements, and a histogram to count map hits.
|
|
||||||
//
|
|
||||||
// Func behavior:
|
|
||||||
//
|
|
||||||
// If the input key is NOT found in the map, the key is wrapped up as
|
|
||||||
// as a variable declaration string and returned, e.g. key FOO becomes $(FOO).
|
|
||||||
// This string is presumably put back where it was found, and might get replaced
|
|
||||||
// later.
|
|
||||||
//
|
|
||||||
// If the key is found in the map, the value is returned if it is a primitive
|
|
||||||
// type (string, bool, number), and the hit is counted.
|
|
||||||
//
|
|
||||||
// If it's not a primitive type (e.g. a map, struct, func, etc.) then this
|
|
||||||
// function doesn't know what to do with it and it returns the key wrapped up
|
|
||||||
// again as if it had not been replaced. This should probably be an error.
|
|
||||||
func MakePrimitiveReplacer(
|
|
||||||
counts map[string]int, someMap map[string]interface{}) MappingFunc {
|
|
||||||
return func(key string) interface{} {
|
|
||||||
if value, ok := someMap[key]; ok {
|
|
||||||
switch typedV := value.(type) {
|
|
||||||
case string, int, int32, int64, float32, float64, bool:
|
|
||||||
counts[key]++
|
|
||||||
return typedV
|
|
||||||
default:
|
|
||||||
// If the value is some complicated type (e.g. a map or struct),
|
|
||||||
// this function doesn't know how to jam it into a string,
|
|
||||||
// so just pretend it was a cache miss.
|
|
||||||
// Likely this should be an error instead of a silent failure,
|
|
||||||
// since the programmer passed an impossible value.
|
|
||||||
log.Printf(
|
|
||||||
"MakePrimitiveReplacer: bad replacement type=%T val=%v",
|
|
||||||
typedV, typedV)
|
|
||||||
return syntaxWrap(key)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If unable to return the mapped variable, return it
|
|
||||||
// as it was found, and a later mapping might be able to
|
|
||||||
// replace it.
|
|
||||||
return syntaxWrap(key)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// DoReplacements replaces variable references in the input string
|
|
||||||
// using the mapping function.
|
|
||||||
func DoReplacements(input string, mapping MappingFunc) interface{} {
|
|
||||||
var buf strings.Builder
|
|
||||||
checkpoint := 0
|
|
||||||
for cursor := 0; cursor < len(input); cursor++ {
|
|
||||||
if input[cursor] == operator && cursor+1 < len(input) {
|
|
||||||
// Copy the portion of the input string since the last
|
|
||||||
// checkpoint into the buffer
|
|
||||||
buf.WriteString(input[checkpoint:cursor])
|
|
||||||
|
|
||||||
// Attempt to read the variable name as defined by the
|
|
||||||
// syntax from the input string
|
|
||||||
read, isVar, advance := tryReadVariableName(input[cursor+1:])
|
|
||||||
|
|
||||||
if isVar {
|
|
||||||
// We were able to read a variable name correctly;
|
|
||||||
// apply the mapping to the variable name and copy the
|
|
||||||
// bytes into the buffer
|
|
||||||
mapped := mapping(read)
|
|
||||||
if input == syntaxWrap(read) {
|
|
||||||
// Preserve the type of variable
|
|
||||||
return mapped
|
|
||||||
}
|
|
||||||
|
|
||||||
// Variable is used in a middle of a string
|
|
||||||
buf.WriteString(fmt.Sprintf("%v", mapped))
|
|
||||||
} else {
|
|
||||||
// Not a variable name; copy the read bytes into the buffer
|
|
||||||
buf.WriteString(read)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Advance the cursor in the input string to account for
|
|
||||||
// bytes consumed to read the variable name expression
|
|
||||||
cursor += advance
|
|
||||||
|
|
||||||
// Advance the checkpoint in the input string
|
|
||||||
checkpoint = cursor + 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the buffer and any remaining unwritten bytes in the
|
|
||||||
// input string.
|
|
||||||
return buf.String() + input[checkpoint:]
|
|
||||||
}
|
|
||||||
|
|
||||||
// tryReadVariableName attempts to read a variable name from the input
|
|
||||||
// string and returns the content read from the input, whether that content
|
|
||||||
// represents a variable name to perform mapping on, and the number of bytes
|
|
||||||
// consumed in the input string.
|
|
||||||
//
|
|
||||||
// The input string is assumed not to contain the initial operator.
|
|
||||||
func tryReadVariableName(input string) (string, bool, int) {
|
|
||||||
switch input[0] {
|
|
||||||
case operator:
|
|
||||||
// Escaped operator; return it.
|
|
||||||
return input[0:1], false, 1
|
|
||||||
case referenceOpener:
|
|
||||||
// Scan to expression closer
|
|
||||||
for i := 1; i < len(input); i++ {
|
|
||||||
if input[i] == referenceCloser {
|
|
||||||
return input[1:i], true, i + 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Incomplete reference; return it.
|
|
||||||
return string(operator) + string(referenceOpener), false, 1
|
|
||||||
default:
|
|
||||||
// Not the beginning of an expression, ie, an operator
|
|
||||||
// that doesn't begin an expression. Return the operator
|
|
||||||
// and the first rune in the string.
|
|
||||||
return string(operator) + string(input[0]), false, 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
package refvar
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter updates $(VAR) style variables with values.
|
|
||||||
// The fieldSpecs are the places to look for occurrences of $(VAR).
|
|
||||||
type Filter struct {
|
|
||||||
MappingFunc MappingFunc `json:"mappingFunc,omitempty" yaml:"mappingFunc,omitempty"`
|
|
||||||
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(f.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
err := node.PipeE(fieldspec.Filter{
|
|
||||||
FieldSpec: f.FieldSpec,
|
|
||||||
SetValue: f.set,
|
|
||||||
})
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) set(node *yaml.RNode) error {
|
|
||||||
if yaml.IsMissingOrNull(node) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
switch node.YNode().Kind {
|
|
||||||
case yaml.ScalarNode:
|
|
||||||
return f.setScalar(node)
|
|
||||||
case yaml.MappingNode:
|
|
||||||
return f.setMap(node)
|
|
||||||
case yaml.SequenceNode:
|
|
||||||
return f.setSeq(node)
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("invalid type encountered %v", node.YNode().Kind)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func updateNodeValue(node *yaml.Node, newValue interface{}) {
|
|
||||||
switch newValue := newValue.(type) {
|
|
||||||
case int:
|
|
||||||
node.Value = strconv.FormatInt(int64(newValue), 10)
|
|
||||||
node.Tag = yaml.NodeTagInt
|
|
||||||
case int32:
|
|
||||||
node.Value = strconv.FormatInt(int64(newValue), 10)
|
|
||||||
node.Tag = yaml.NodeTagInt
|
|
||||||
case int64:
|
|
||||||
node.Value = strconv.FormatInt(newValue, 10)
|
|
||||||
node.Tag = yaml.NodeTagInt
|
|
||||||
case bool:
|
|
||||||
node.SetString(strconv.FormatBool(newValue))
|
|
||||||
node.Tag = yaml.NodeTagBool
|
|
||||||
case float32:
|
|
||||||
node.SetString(strconv.FormatFloat(float64(newValue), 'f', -1, 32))
|
|
||||||
node.Tag = yaml.NodeTagFloat
|
|
||||||
case float64:
|
|
||||||
node.SetString(strconv.FormatFloat(newValue, 'f', -1, 64))
|
|
||||||
node.Tag = yaml.NodeTagFloat
|
|
||||||
default:
|
|
||||||
node.SetString(newValue.(string))
|
|
||||||
node.Tag = yaml.NodeTagString
|
|
||||||
}
|
|
||||||
node.Style = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) setScalar(node *yaml.RNode) error {
|
|
||||||
if !yaml.IsYNodeString(node.YNode()) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
v := DoReplacements(node.YNode().Value, f.MappingFunc)
|
|
||||||
updateNodeValue(node.YNode(), v)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) setMap(node *yaml.RNode) error {
|
|
||||||
contents := node.YNode().Content
|
|
||||||
for i := 0; i < len(contents); i += 2 {
|
|
||||||
if !yaml.IsYNodeString(contents[i]) {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"invalid map key: value='%s', tag='%s'",
|
|
||||||
contents[i].Value, contents[i].Tag)
|
|
||||||
}
|
|
||||||
if !yaml.IsYNodeString(contents[i+1]) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
newValue := DoReplacements(contents[i+1].Value, f.MappingFunc)
|
|
||||||
updateNodeValue(contents[i+1], newValue)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f Filter) setSeq(node *yaml.RNode) error {
|
|
||||||
for _, item := range node.YNode().Content {
|
|
||||||
if !yaml.IsYNodeString(item) {
|
|
||||||
return fmt.Errorf("invalid value type expect a string")
|
|
||||||
}
|
|
||||||
newValue := DoReplacements(item.Value, f.MappingFunc)
|
|
||||||
updateNodeValue(item, newValue)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -1,299 +0,0 @@
|
|||||||
package refvar_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
. "sigs.k8s.io/kustomize/api/filters/refvar"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
var makeMf = func(theMap map[string]interface{}) MappingFunc {
|
|
||||||
ignored := make(map[string]int)
|
|
||||||
return MakePrimitiveReplacer(ignored, theMap)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter Filter
|
|
||||||
}{
|
|
||||||
"simple scalar": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: $(VAR)`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 5`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"VAR": int64(5),
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"non-string scalar": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 1`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 1`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"VAR": int64(5),
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"wrong path": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 1`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 1`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"VAR": int64(5),
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "a/b/c"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sequence": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
- $(FOO)
|
|
||||||
- $(BAR)
|
|
||||||
- $(BAZ)
|
|
||||||
- $(FOO)+$(BAR)
|
|
||||||
- $(BOOL)
|
|
||||||
- $(FLOAT)`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
- foo
|
|
||||||
- bar
|
|
||||||
- $(BAZ)
|
|
||||||
- foo+bar
|
|
||||||
- false
|
|
||||||
- 1.23`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"FOO": "foo",
|
|
||||||
"BAR": "bar",
|
|
||||||
"BOOL": false,
|
|
||||||
"FLOAT": 1.23,
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"maps": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
FOO: $(FOO)
|
|
||||||
BAR: $(BAR)
|
|
||||||
BAZ: $(BAZ)
|
|
||||||
PLUS: $(FOO)+$(BAR)`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
FOO: foo
|
|
||||||
BAR: bar
|
|
||||||
BAZ: $(BAZ)
|
|
||||||
PLUS: foo+bar`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"FOO": "foo",
|
|
||||||
"BAR": "bar",
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"complicated case": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
slice1:
|
|
||||||
- $(FOO)
|
|
||||||
slice2:
|
|
||||||
FOO: $(FOO)
|
|
||||||
BAR: $(BAR)
|
|
||||||
BOOL: false
|
|
||||||
INT: 0
|
|
||||||
SLICE:
|
|
||||||
- $(FOO)`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
slice1:
|
|
||||||
- $(FOO)
|
|
||||||
slice2:
|
|
||||||
FOO: foo
|
|
||||||
BAR: bar
|
|
||||||
BOOL: false
|
|
||||||
INT: 0
|
|
||||||
SLICE:
|
|
||||||
- $(FOO)`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"FOO": "foo",
|
|
||||||
"BAR": "bar",
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data/slice2"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"null value": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
FOO: null`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
FOO: null`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
// no replacements!
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data/FOO"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expected),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest_test.RunFilter(t, tc.input, tc.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFilterUnhappy(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedError string
|
|
||||||
filter Filter
|
|
||||||
}{
|
|
||||||
"non-string in sequence": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
slice:
|
|
||||||
- false`,
|
|
||||||
expectedError: `considering field 'data/slice' of object
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
annotations:
|
|
||||||
config.kubernetes.io/index: '0'
|
|
||||||
data:
|
|
||||||
slice:
|
|
||||||
- false
|
|
||||||
: invalid value type expect a string`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"VAR": int64(5),
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data/slice"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"invalid key in map": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
data:
|
|
||||||
1: str`,
|
|
||||||
expectedError: `considering field 'data' of object
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
annotations:
|
|
||||||
config.kubernetes.io/index: '0'
|
|
||||||
data:
|
|
||||||
1: str
|
|
||||||
: invalid map key: value='1', tag='` + yaml.NodeTagInt + `'`,
|
|
||||||
filter: Filter{
|
|
||||||
MappingFunc: makeMf(map[string]interface{}{
|
|
||||||
"VAR": int64(5),
|
|
||||||
}),
|
|
||||||
FieldSpec: types.FieldSpec{Path: "data"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
_, err := filtertest_test.RunFilterE(t, tc.input, tc.filter)
|
|
||||||
if !assert.EqualError(t, err, tc.expectedError) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
// Package replicacount contains a kio.Filter implementation of the kustomize
|
|
||||||
// ReplicaCountTransformer.
|
|
||||||
package replicacount
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
package replicacount
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
)
|
|
||||||
|
|
||||||
func ExampleFilter() {
|
|
||||||
err := kio.Pipeline{
|
|
||||||
Inputs: []kio.Reader{&kio.ByteReader{Reader: bytes.NewBufferString(`
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Foo
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
replicas: 5
|
|
||||||
---
|
|
||||||
apiVersion: example.com/v1
|
|
||||||
kind: Bar
|
|
||||||
metadata:
|
|
||||||
name: instance
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
replicas: 5
|
|
||||||
`)}},
|
|
||||||
Filters: []kio.Filter{Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Count: 42,
|
|
||||||
Name: "instance",
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{
|
|
||||||
Path: "spec/template/replicas",
|
|
||||||
},
|
|
||||||
}},
|
|
||||||
Outputs: []kio.Writer{kio.ByteWriter{Writer: os.Stdout}},
|
|
||||||
}.Execute()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Output:
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Foo
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// template:
|
|
||||||
// replicas: 42
|
|
||||||
// ---
|
|
||||||
// apiVersion: example.com/v1
|
|
||||||
// kind: Bar
|
|
||||||
// metadata:
|
|
||||||
// name: instance
|
|
||||||
// spec:
|
|
||||||
// template:
|
|
||||||
// replicas: 42
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
package replicacount
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/fieldspec"
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/filtersutil"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Filter updates/sets replicas fields using the fieldSpecs
|
|
||||||
type Filter struct {
|
|
||||||
Replica types.Replica `json:"replica,omitempty" yaml:"replica,omitempty"`
|
|
||||||
FieldSpec types.FieldSpec `json:"fieldSpec,omitempty" yaml:"fieldSpec,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (rc Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
return kio.FilterAll(yaml.FilterFunc(rc.run)).Filter(nodes)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (rc Filter) run(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
err := node.PipeE(fieldspec.Filter{
|
|
||||||
FieldSpec: rc.FieldSpec,
|
|
||||||
SetValue: rc.set,
|
|
||||||
CreateKind: yaml.ScalarNode, // replicas is a ScalarNode
|
|
||||||
CreateTag: yaml.NodeTagInt,
|
|
||||||
})
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (rc Filter) set(node *yaml.RNode) error {
|
|
||||||
return filtersutil.SetScalar(strconv.FormatInt(rc.Replica.Count, 10))(node)
|
|
||||||
}
|
|
||||||
@@ -1,176 +0,0 @@
|
|||||||
package replicacount
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestFilter(t *testing.T) {
|
|
||||||
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expected string
|
|
||||||
filter Filter
|
|
||||||
}{
|
|
||||||
"update field": {
|
|
||||||
input: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 5
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: dep
|
|
||||||
spec:
|
|
||||||
replicas: 42
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Name: "dep",
|
|
||||||
Count: 42,
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/replicas"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"add field": {
|
|
||||||
input: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
replicas: 42
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Name: "cus",
|
|
||||||
Count: 42,
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{
|
|
||||||
Path: "spec/template/replicas",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
"add_field_null": {
|
|
||||||
input: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
replicas: null
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
replicas: 42
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Name: "cus",
|
|
||||||
Count: 42,
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{
|
|
||||||
Path: "spec/template/replicas",
|
|
||||||
CreateIfNotPresent: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"no update if CreateIfNotPresent is false": {
|
|
||||||
input: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
other: something
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Name: "cus",
|
|
||||||
Count: 42,
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{
|
|
||||||
Path: "spec/template/replicas",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update multiple fields": {
|
|
||||||
input: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
replicas: 5
|
|
||||||
`,
|
|
||||||
expected: `
|
|
||||||
apiVersion: custom/v1
|
|
||||||
kind: Custom
|
|
||||||
metadata:
|
|
||||||
name: cus
|
|
||||||
spec:
|
|
||||||
template:
|
|
||||||
replicas: 42
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Replica: types.Replica{
|
|
||||||
Name: "cus",
|
|
||||||
Count: 42,
|
|
||||||
},
|
|
||||||
FieldSpec: types.FieldSpec{Path: "spec/template/replicas"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expected),
|
|
||||||
strings.TrimSpace(
|
|
||||||
filtertest_test.RunFilter(t, tc.input, tc.filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package valueadd
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/kio"
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
|
||||||
|
|
||||||
// An 'Add' operation aspiring to IETF RFC 6902 JSON.
|
|
||||||
//
|
|
||||||
// The filter tries to add a value to a node at a particular field path.
|
|
||||||
//
|
|
||||||
// Kinds of target fields:
|
|
||||||
//
|
|
||||||
// - Non-existent target field.
|
|
||||||
//
|
|
||||||
// The field will be added and the value inserted.
|
|
||||||
//
|
|
||||||
// - Existing field, scalar or map.
|
|
||||||
//
|
|
||||||
// E.g. 'spec/template/spec/containers/[name:nginx]/image'
|
|
||||||
//
|
|
||||||
// This behaves like an IETF RFC 6902 Replace operation would;
|
|
||||||
// the existing value is replaced without complaint, even though
|
|
||||||
// this is an Add operation. In contrast, a Replace operation
|
|
||||||
// must fail (report an error) if the field doesn't exist.
|
|
||||||
//
|
|
||||||
// - Existing field, list (array)
|
|
||||||
// Not supported yet.
|
|
||||||
// TODO: Honor fields with RFC-6902-style array indices
|
|
||||||
// TODO: like 'spec/template/spec/containers/2'
|
|
||||||
// TODO: Modify kyaml/yaml/PathGetter to allow this.
|
|
||||||
// The value will be inserted into the array at the given position,
|
|
||||||
// shifting other contents. To instead replace an array entry, use
|
|
||||||
// an implementation of an IETF RFC 6902 Replace operation.
|
|
||||||
//
|
|
||||||
// For the common case of a filepath in the field value, and a desire
|
|
||||||
// to add the value to the filepath (rather than replace the filepath),
|
|
||||||
// use a non-zero value of FilePathPosition (see below).
|
|
||||||
type Filter struct {
|
|
||||||
// Value is the value to add.
|
|
||||||
//
|
|
||||||
// Empty values are disallowed, i.e. this filter isn't intended
|
|
||||||
// for use in erasing or removing fields. For that, use a filter
|
|
||||||
// more aligned with the IETF RFC 6902 JSON Remove operation.
|
|
||||||
//
|
|
||||||
// At the time of writing, Value's value should be a simple string,
|
|
||||||
// not a JSON document. This particular filter focuses on easing
|
|
||||||
// injection of a single-sourced cloud project and/or cluster name
|
|
||||||
// into various fields, especially namespace and various filepath
|
|
||||||
// specifications.
|
|
||||||
Value string
|
|
||||||
|
|
||||||
// FieldPath is a JSON-style path to the field intended to hold the value.
|
|
||||||
FieldPath string
|
|
||||||
|
|
||||||
// FilePathPosition is a filepath field index.
|
|
||||||
//
|
|
||||||
// Call the value of this field _i_.
|
|
||||||
//
|
|
||||||
// If _i_ is zero, negative or unspecified, this field has no effect.
|
|
||||||
//
|
|
||||||
// If _i_ is > 0, then it's assumed that
|
|
||||||
// - 'Value' is a string that can work as a directory or file name,
|
|
||||||
// - the field value intended for replacement holds a filepath.
|
|
||||||
//
|
|
||||||
// The filepath is split into a string slice, the value is inserted
|
|
||||||
// at position [i-1], shifting the rest of the path to the right.
|
|
||||||
// A value of i==1 puts the new value at the start of the path.
|
|
||||||
// This change never converts an absolute path to a relative path,
|
|
||||||
// meaning adding a new field at position i==1 will preserve a
|
|
||||||
// leading slash. E.g. if Value == 'PEACH'
|
|
||||||
//
|
|
||||||
// OLD : NEW : FilePathPosition
|
|
||||||
// --------------------------------------------------------
|
|
||||||
// {empty} : PEACH : irrelevant
|
|
||||||
// / : /PEACH : irrelevant
|
|
||||||
// pie : PEACH/pie : 1 (or less to prefix)
|
|
||||||
// /pie : /PEACH/pie : 1 (or less to prefix)
|
|
||||||
// raw : raw/PEACH : 2 (or more to postfix)
|
|
||||||
// /raw : /raw/PEACH : 2 (or more to postfix)
|
|
||||||
// a/nice/warm/pie : a/nice/warm/PEACH/pie : 4
|
|
||||||
// /a/nice/warm/pie : /a/nice/warm/PEACH/pie : 4
|
|
||||||
//
|
|
||||||
// For robustness (liberal input, conservative output) FilePathPosition
|
|
||||||
// values that that are too large to index the split filepath result in a
|
|
||||||
// postfix rather than an error. So use 1 to prefix, 9999 to postfix.
|
|
||||||
FilePathPosition int `json:"filePathPosition,omitempty" yaml:"filePathPosition,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ kio.Filter = Filter{}
|
|
||||||
|
|
||||||
func (f Filter) Filter(nodes []*yaml.RNode) ([]*yaml.RNode, error) {
|
|
||||||
_, err := kio.FilterAll(yaml.FilterFunc(
|
|
||||||
func(node *yaml.RNode) (*yaml.RNode, error) {
|
|
||||||
var fields []string
|
|
||||||
// if there is forward slash '/' in the field name, a back slash '\'
|
|
||||||
// will be used to escape it.
|
|
||||||
for _, f := range strings.Split(f.FieldPath, "/") {
|
|
||||||
if len(fields) > 0 && strings.HasSuffix(fields[len(fields)-1], "\\") {
|
|
||||||
concatField := strings.TrimSuffix(fields[len(fields)-1], "\\") + "/" + f
|
|
||||||
fields = append(fields[:len(fields)-1], concatField)
|
|
||||||
} else {
|
|
||||||
fields = append(fields, f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO: support SequenceNode.
|
|
||||||
// Presumably here one could look for array indices (digits) at
|
|
||||||
// the end of the field path (as described in IETF RFC 6902 JSON),
|
|
||||||
// and if found, take it as a signal that this should be a
|
|
||||||
// SequenceNode instead of a ScalarNode, and insert the value
|
|
||||||
// into the proper slot, shifting every over.
|
|
||||||
n, err := node.Pipe(yaml.LookupCreate(yaml.ScalarNode, fields...))
|
|
||||||
if err != nil {
|
|
||||||
return node, err
|
|
||||||
}
|
|
||||||
// TODO: allow more kinds
|
|
||||||
if err := yaml.ErrorIfInvalid(n, yaml.ScalarNode); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
newValue := f.Value
|
|
||||||
if f.FilePathPosition > 0 {
|
|
||||||
newValue = filesys.InsertPathPart(
|
|
||||||
n.YNode().Value, f.FilePathPosition-1, newValue)
|
|
||||||
}
|
|
||||||
return n.Pipe(yaml.FieldSetter{StringValue: newValue})
|
|
||||||
})).Filter(nodes)
|
|
||||||
return nodes, err
|
|
||||||
}
|
|
||||||
@@ -1,123 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package valueadd
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
filtertest_test "sigs.k8s.io/kustomize/api/testutils/filtertest"
|
|
||||||
)
|
|
||||||
|
|
||||||
const someResource = `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: projects/whatever
|
|
||||||
`
|
|
||||||
|
|
||||||
func TestValueAddFilter(t *testing.T) {
|
|
||||||
testCases := map[string]struct {
|
|
||||||
input string
|
|
||||||
expectedOutput string
|
|
||||||
filter Filter
|
|
||||||
}{
|
|
||||||
"simpleAdd": {
|
|
||||||
input: `
|
|
||||||
kind: SomeKind
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: valueAdded
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef/external",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"replaceExisting": {
|
|
||||||
input: someResource,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: valueAdded
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef/external",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"prefixExisting": {
|
|
||||||
input: someResource,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: valueAdded/projects/whatever
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef/external",
|
|
||||||
FilePathPosition: 1,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"postfixExisting": {
|
|
||||||
input: someResource,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: projects/whatever/valueAdded
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef/external",
|
|
||||||
FilePathPosition: 99,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"placeInMiddleOfExisting": {
|
|
||||||
input: someResource,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef:
|
|
||||||
external: projects/valueAdded/whatever
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef/external",
|
|
||||||
FilePathPosition: 2,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"backSlash": {
|
|
||||||
input: `
|
|
||||||
kind: SomeKind
|
|
||||||
`,
|
|
||||||
expectedOutput: `
|
|
||||||
kind: SomeKind
|
|
||||||
spec:
|
|
||||||
resourceRef/external: valueAdded
|
|
||||||
`,
|
|
||||||
filter: Filter{
|
|
||||||
Value: "valueAdded",
|
|
||||||
FieldPath: "spec/resourceRef\\/external",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for tn, tc := range testCases {
|
|
||||||
t.Run(tn, func(t *testing.T) {
|
|
||||||
filter := tc.filter
|
|
||||||
if !assert.Equal(t,
|
|
||||||
strings.TrimSpace(tc.expectedOutput),
|
|
||||||
strings.TrimSpace(filtertest_test.RunFilter(t, tc.input, filter))) {
|
|
||||||
t.FailNow()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
19
api/go.mod
19
api/go.mod
@@ -1,25 +1,18 @@
|
|||||||
module sigs.k8s.io/kustomize/api
|
module sigs.k8s.io/kustomize/api
|
||||||
|
|
||||||
go 1.15
|
go 1.13
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/evanphx/json-patch v4.5.0+incompatible
|
github.com/evanphx/json-patch v4.5.0+incompatible
|
||||||
github.com/go-errors/errors v1.0.1
|
github.com/go-openapi/spec v0.19.4
|
||||||
github.com/go-openapi/spec v0.19.5
|
|
||||||
github.com/golangci/golangci-lint v1.21.0
|
github.com/golangci/golangci-lint v1.21.0
|
||||||
github.com/google/go-cmp v0.3.0
|
|
||||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510
|
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510
|
||||||
github.com/hashicorp/go-multierror v1.1.0
|
|
||||||
github.com/imdario/mergo v0.3.5
|
|
||||||
github.com/pkg/errors v0.8.1
|
github.com/pkg/errors v0.8.1
|
||||||
github.com/stretchr/testify v1.4.0
|
golang.org/x/tools v0.0.0-20191010075000-0337d82405ff
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf
|
gopkg.in/yaml.v2 v2.2.4
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e
|
|
||||||
gopkg.in/yaml.v2 v2.3.0
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c
|
|
||||||
k8s.io/api v0.17.0
|
k8s.io/api v0.17.0
|
||||||
k8s.io/apimachinery v0.17.0
|
k8s.io/apimachinery v0.17.0
|
||||||
k8s.io/client-go v0.17.0
|
k8s.io/client-go v0.17.0
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.9
|
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a
|
||||||
sigs.k8s.io/yaml v1.2.0
|
sigs.k8s.io/yaml v1.1.0
|
||||||
)
|
)
|
||||||
|
|||||||
148
api/go.sum
148
api/go.sum
@@ -1,7 +1,6 @@
|
|||||||
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
||||||
github.com/360EntSecGroup-Skylar/excelize v1.4.1/go.mod h1:vnax29X2usfl7HHkBrX5EvSCJcmH3dT9luvxzu8iGAE=
|
|
||||||
github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI=
|
github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI=
|
||||||
github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0=
|
github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0=
|
||||||
github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA=
|
github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA=
|
||||||
@@ -15,33 +14,21 @@ github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb0
|
|||||||
github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
|
github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
|
||||||
github.com/OpenPeeDeeP/depguard v1.0.1 h1:VlW4R6jmBIv3/u1JNlawEvJMM4J+dPORPaZasQee8Us=
|
github.com/OpenPeeDeeP/depguard v1.0.1 h1:VlW4R6jmBIv3/u1JNlawEvJMM4J+dPORPaZasQee8Us=
|
||||||
github.com/OpenPeeDeeP/depguard v1.0.1/go.mod h1:xsIw86fROiiwelg+jB2uM9PiKihMMmUx/1V+TNhjQvM=
|
github.com/OpenPeeDeeP/depguard v1.0.1/go.mod h1:xsIw86fROiiwelg+jB2uM9PiKihMMmUx/1V+TNhjQvM=
|
||||||
github.com/PuerkitoBio/goquery v1.5.0/go.mod h1:qD2PgZ9lccMbQlc7eEOjaeRlFQON7xY8kdmcsrnKqMg=
|
|
||||||
github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
||||||
github.com/PuerkitoBio/purell v1.1.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
|
||||||
github.com/PuerkitoBio/purell v1.1.1 h1:WEQqlqaGbrPkxLJWfBwQmfEAE1Z7ONdDLqrN38tNFfI=
|
github.com/PuerkitoBio/purell v1.1.1 h1:WEQqlqaGbrPkxLJWfBwQmfEAE1Z7ONdDLqrN38tNFfI=
|
||||||
github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
github.com/PuerkitoBio/purell v1.1.1/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
||||||
github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
|
github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
|
||||||
github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 h1:d+Bc7a5rLufV/sSk/8dngufqelfh6jnri85riMAaF/M=
|
github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578 h1:d+Bc7a5rLufV/sSk/8dngufqelfh6jnri85riMAaF/M=
|
||||||
github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
|
github.com/PuerkitoBio/urlesc v0.0.0-20170810143723-de5bf2ad4578/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
|
||||||
github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6/go.mod h1:3eOhrUMpNV+6aFIbp5/iudMxNCF27Vw2OZgy4xEx0Fg=
|
github.com/StackExchange/wmi v0.0.0-20180116203802-5d049714c4a6/go.mod h1:3eOhrUMpNV+6aFIbp5/iudMxNCF27Vw2OZgy4xEx0Fg=
|
||||||
github.com/agnivade/levenshtein v1.0.1/go.mod h1:CURSv5d9Uaml+FovSIICkLbAUZ9S4RqaHDIsdSBg7lM=
|
|
||||||
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
||||||
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
||||||
github.com/andreyvit/diff v0.0.0-20170406064948-c7f18ee00883/go.mod h1:rCTlJbsFo29Kk6CurOXKm700vrz8f0KW0JNfpkRJY/8=
|
|
||||||
github.com/andybalholm/cascadia v1.0.0/go.mod h1:GsXiBklL0woXo1j/WYWtSYYC4ouU9PqHO0sqidkEA4Y=
|
|
||||||
github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8=
|
github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8=
|
||||||
github.com/asaskevich/govalidator v0.0.0-20180720115003-f9ffefc3facf/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY=
|
|
||||||
github.com/asaskevich/govalidator v0.0.0-20190424111038-f61b66f89f4a/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY=
|
|
||||||
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
|
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
|
||||||
github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
|
github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
|
||||||
github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d h1:xDfNPAt8lFiC1UJrqV3uuy861HCTo708pDMbjHHdCas=
|
|
||||||
github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d/go.mod h1:6QX/PXZ00z/TKoufEY6K/a0k6AhaJrQKdFe6OfVXsa4=
|
|
||||||
github.com/bombsimon/wsl v1.2.5 h1:9gTOkIwVtoDZywvX802SDHokeX4kW1cKnV8ZTVAPkRs=
|
github.com/bombsimon/wsl v1.2.5 h1:9gTOkIwVtoDZywvX802SDHokeX4kW1cKnV8ZTVAPkRs=
|
||||||
github.com/bombsimon/wsl v1.2.5/go.mod h1:43lEF/i0kpXbLCeDXL9LMT8c92HyBywXb0AsgMHYngM=
|
github.com/bombsimon/wsl v1.2.5/go.mod h1:43lEF/i0kpXbLCeDXL9LMT8c92HyBywXb0AsgMHYngM=
|
||||||
github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
|
github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
|
||||||
github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI=
|
|
||||||
github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI=
|
|
||||||
github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU=
|
|
||||||
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
||||||
github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk=
|
github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk=
|
||||||
github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
|
github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
|
||||||
@@ -50,7 +37,6 @@ github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3Ee
|
|||||||
github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
|
github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
|
||||||
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
|
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
|
||||||
github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE=
|
github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE=
|
||||||
github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
|
|
||||||
github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY=
|
github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY=
|
||||||
github.com/davecgh/go-spew v0.0.0-20151105211317-5215b55f46b2/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v0.0.0-20151105211317-5215b55f46b2/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
@@ -58,10 +44,7 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
|
|||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
|
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
|
||||||
github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no=
|
github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no=
|
||||||
github.com/docker/go-units v0.3.3/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
|
||||||
github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
|
||||||
github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM=
|
github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM=
|
||||||
github.com/dustmop/soup v1.1.2-0.20190516214245-38228baa104e/go.mod h1:CgNC6SGbT+Xb8wGGvzilttZL1mc5sQ/5KkcxsZttMIk=
|
|
||||||
github.com/elazarl/goproxy v0.0.0-20170405201442-c4fc26588b6e/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
|
github.com/elazarl/goproxy v0.0.0-20170405201442-c4fc26588b6e/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
|
||||||
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633 h1:H2pdYOb3KQ1/YsqVWoWNLQO+fusocsw354rqGTZtAgw=
|
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633 h1:H2pdYOb3KQ1/YsqVWoWNLQO+fusocsw354rqGTZtAgw=
|
||||||
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
||||||
@@ -74,12 +57,8 @@ github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV
|
|||||||
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
||||||
github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
||||||
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
||||||
github.com/globalsign/mgo v0.0.0-20180905125535-1ca0a4f7cbcb/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q=
|
|
||||||
github.com/globalsign/mgo v0.0.0-20181015135952-eeefdecb41b8/go.mod h1:xkRDCp4j0OGD1HRkm4kmhM+pmpv3AKq5SU7GMg4oO/Q=
|
|
||||||
github.com/go-critic/go-critic v0.3.5-0.20190904082202-d79a9f0c64db h1:GYXWx7Vr3+zv833u+8IoXbNnQY0AdXsxAgI0kX7xcwA=
|
github.com/go-critic/go-critic v0.3.5-0.20190904082202-d79a9f0c64db h1:GYXWx7Vr3+zv833u+8IoXbNnQY0AdXsxAgI0kX7xcwA=
|
||||||
github.com/go-critic/go-critic v0.3.5-0.20190904082202-d79a9f0c64db/go.mod h1:+sE8vrLDS2M0pZkBk0wy6+nLdKexVDrl/jBqQOTDThA=
|
github.com/go-critic/go-critic v0.3.5-0.20190904082202-d79a9f0c64db/go.mod h1:+sE8vrLDS2M0pZkBk0wy6+nLdKexVDrl/jBqQOTDThA=
|
||||||
github.com/go-errors/errors v1.0.1 h1:LUHzmkK3GUKUrL/1gfBUxAHzcev3apQlezX/+O7ma6w=
|
|
||||||
github.com/go-errors/errors v1.0.1/go.mod h1:f4zRHt4oKfwPJE5k8C9vpYG+aDHdBFUsgrm6/TyX73Q=
|
|
||||||
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
||||||
github.com/go-lintpack/lintpack v0.5.2 h1:DI5mA3+eKdWeJ40nU4d6Wc26qmdG8RCi/btYq0TuRN0=
|
github.com/go-lintpack/lintpack v0.5.2 h1:DI5mA3+eKdWeJ40nU4d6Wc26qmdG8RCi/btYq0TuRN0=
|
||||||
github.com/go-lintpack/lintpack v0.5.2/go.mod h1:NwZuYi2nUHho8XEIZ6SIxihrnPoqBTDqfpXvXAN0sXM=
|
github.com/go-lintpack/lintpack v0.5.2/go.mod h1:NwZuYi2nUHho8XEIZ6SIxihrnPoqBTDqfpXvXAN0sXM=
|
||||||
@@ -87,55 +66,20 @@ github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9
|
|||||||
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
||||||
github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas=
|
github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas=
|
||||||
github.com/go-ole/go-ole v1.2.1/go.mod h1:7FAglXiTm7HKlQRDeOQ6ZNUHidzCWXuZWq/1dTyBNF8=
|
github.com/go-ole/go-ole v1.2.1/go.mod h1:7FAglXiTm7HKlQRDeOQ6ZNUHidzCWXuZWq/1dTyBNF8=
|
||||||
github.com/go-openapi/analysis v0.0.0-20180825180245-b006789cd277/go.mod h1:k70tL6pCuVxPJOHXQ+wIac1FUrvNkHolPie/cLEU6hI=
|
|
||||||
github.com/go-openapi/analysis v0.17.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
|
|
||||||
github.com/go-openapi/analysis v0.18.0/go.mod h1:IowGgpVeD0vNm45So8nr+IcQ3pxVtpRoBWb8PVZO0ik=
|
|
||||||
github.com/go-openapi/analysis v0.19.2/go.mod h1:3P1osvZa9jKjb8ed2TPng3f0i/UY9snX6gxi44djMjk=
|
|
||||||
github.com/go-openapi/analysis v0.19.5/go.mod h1:hkEAkxagaIvIP7VTn8ygJNkd4kAYON2rCu0v0ObL0AU=
|
|
||||||
github.com/go-openapi/errors v0.17.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0=
|
|
||||||
github.com/go-openapi/errors v0.18.0/go.mod h1:LcZQpmvG4wyF5j4IhA73wkLFQg+QJXOQHVjmcZxhka0=
|
|
||||||
github.com/go-openapi/errors v0.19.2/go.mod h1:qX0BLWsyaKfvhluLejVpVNwNRdXZhEbTA4kxxpKBC94=
|
|
||||||
github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0=
|
github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0=
|
||||||
github.com/go-openapi/jsonpointer v0.17.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M=
|
|
||||||
github.com/go-openapi/jsonpointer v0.18.0/go.mod h1:cOnomiV+CVVwFLk0A/MExoFMjwdsUdVpsRhURCKh+3M=
|
|
||||||
github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg=
|
github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg=
|
||||||
github.com/go-openapi/jsonpointer v0.19.3 h1:gihV7YNZK1iK6Tgwwsxo2rJbD1GTbdm72325Bq8FI3w=
|
github.com/go-openapi/jsonpointer v0.19.3 h1:gihV7YNZK1iK6Tgwwsxo2rJbD1GTbdm72325Bq8FI3w=
|
||||||
github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg=
|
github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg=
|
||||||
github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg=
|
github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg=
|
||||||
github.com/go-openapi/jsonreference v0.17.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
|
github.com/go-openapi/jsonreference v0.19.2 h1:o20suLFB4Ri0tuzpWtyHlh7E7HnkqTNLq6aR6WVNS1w=
|
||||||
github.com/go-openapi/jsonreference v0.18.0/go.mod h1:g4xxGn04lDIRh0GJb5QlpE3HfopLOL6uZrK/VgnsK9I=
|
|
||||||
github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc=
|
github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc=
|
||||||
github.com/go-openapi/jsonreference v0.19.3 h1:5cxNfTy0UVC3X8JL5ymxzyoUZmo8iZb+jeTWn7tUa8o=
|
|
||||||
github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8=
|
|
||||||
github.com/go-openapi/loads v0.17.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
|
|
||||||
github.com/go-openapi/loads v0.18.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
|
|
||||||
github.com/go-openapi/loads v0.19.0/go.mod h1:72tmFy5wsWx89uEVddd0RjRWPZm92WRLhf7AC+0+OOU=
|
|
||||||
github.com/go-openapi/loads v0.19.2/go.mod h1:QAskZPMX5V0C2gvfkGZzJlINuP7Hx/4+ix5jWFxsNPs=
|
|
||||||
github.com/go-openapi/loads v0.19.4/go.mod h1:zZVHonKd8DXyxyw4yfnVjPzBjIQcLt0CCsn0N0ZrQsk=
|
|
||||||
github.com/go-openapi/runtime v0.0.0-20180920151709-4f900dc2ade9/go.mod h1:6v9a6LTXWQCdL8k1AO3cvqx5OtZY/Y9wKTgaoP6YRfA=
|
|
||||||
github.com/go-openapi/runtime v0.19.0/go.mod h1:OwNfisksmmaZse4+gpV3Ne9AyMOlP1lt4sK4FXt0O64=
|
|
||||||
github.com/go-openapi/runtime v0.19.4/go.mod h1:X277bwSUBxVlCYR3r7xgZZGKVvBd/29gLDlFGtJ8NL4=
|
|
||||||
github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc=
|
github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc=
|
||||||
github.com/go-openapi/spec v0.17.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI=
|
github.com/go-openapi/spec v0.19.4 h1:ixzUSnHTd6hCemgtAJgluaTSGYpLNpJY4mA2DIkdOAo=
|
||||||
github.com/go-openapi/spec v0.18.0/go.mod h1:XkF/MOi14NmjsfZ8VtAKf8pIlbZzyoTvZsdfssdxcBI=
|
github.com/go-openapi/spec v0.19.4/go.mod h1:FpwSN1ksY1eteniUU7X0N/BgJ7a4WvBFVA8Lj9mJglo=
|
||||||
github.com/go-openapi/spec v0.19.2/go.mod h1:sCxk3jxKgioEJikev4fgkNmwS+3kuYdJtcsZsD5zxMY=
|
|
||||||
github.com/go-openapi/spec v0.19.3/go.mod h1:FpwSN1ksY1eteniUU7X0N/BgJ7a4WvBFVA8Lj9mJglo=
|
|
||||||
github.com/go-openapi/spec v0.19.5 h1:Xm0Ao53uqnk9QE/LlYV5DEU09UAgpliA85QoT9LzqPw=
|
|
||||||
github.com/go-openapi/spec v0.19.5/go.mod h1:Hm2Jr4jv8G1ciIAo+frC/Ft+rR2kQDh8JHKHb3gWUSk=
|
|
||||||
github.com/go-openapi/strfmt v0.17.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU=
|
|
||||||
github.com/go-openapi/strfmt v0.18.0/go.mod h1:P82hnJI0CXkErkXi8IKjPbNBM6lV6+5pLP5l494TcyU=
|
|
||||||
github.com/go-openapi/strfmt v0.19.0/go.mod h1:+uW+93UVvGGq2qGaZxdDeJqSAqBqBdl+ZPMF/cC8nDY=
|
|
||||||
github.com/go-openapi/strfmt v0.19.3/go.mod h1:0yX7dbo8mKIvc3XSKp7MNfxw4JytCfCD6+bY1AVL9LU=
|
|
||||||
github.com/go-openapi/strfmt v0.19.5/go.mod h1:eftuHTlB/dI8Uq8JJOyRlieZf+WkkxUuk0dgdHXr2Qk=
|
|
||||||
github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I=
|
github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I=
|
||||||
github.com/go-openapi/swag v0.17.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg=
|
|
||||||
github.com/go-openapi/swag v0.18.0/go.mod h1:AByQ+nYG6gQg71GINrmuDXCPWdL640yX49/kXLo40Tg=
|
|
||||||
github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
|
github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
|
||||||
github.com/go-openapi/swag v0.19.5 h1:lTz6Ys4CmqqCQmZPBlbQENR1/GucA2bzYTE12Pw4tFY=
|
github.com/go-openapi/swag v0.19.5 h1:lTz6Ys4CmqqCQmZPBlbQENR1/GucA2bzYTE12Pw4tFY=
|
||||||
github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
|
github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
|
||||||
github.com/go-openapi/validate v0.18.0/go.mod h1:Uh4HdOzKt19xGIGm1qHf/ofbX1YQ4Y+MYsct2VUrAJ4=
|
|
||||||
github.com/go-openapi/validate v0.19.2/go.mod h1:1tRCw7m3jtI8eNWEEliiAqUIcBztB2KDnRCRMUi7GTA=
|
|
||||||
github.com/go-openapi/validate v0.19.8/go.mod h1:8DJv2CVJQ6kGNpFW6eV9N3JviE1C85nY1c2z52x1Gk4=
|
|
||||||
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
|
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
|
||||||
github.com/go-toolsmith/astcast v1.0.0 h1:JojxlmI6STnFVG9yOImLeGREv8W2ocNUM+iOhR6jE7g=
|
github.com/go-toolsmith/astcast v1.0.0 h1:JojxlmI6STnFVG9yOImLeGREv8W2ocNUM+iOhR6jE7g=
|
||||||
github.com/go-toolsmith/astcast v1.0.0/go.mod h1:mt2OdQTeAQcY4DQgPSArJjHCcOwlX+Wl/kwN+LbLGQ4=
|
github.com/go-toolsmith/astcast v1.0.0/go.mod h1:mt2OdQTeAQcY4DQgPSArJjHCcOwlX+Wl/kwN+LbLGQ4=
|
||||||
@@ -158,7 +102,6 @@ github.com/go-toolsmith/strparse v1.0.0 h1:Vcw78DnpCAKlM20kSbAyO4mPfJn/lyYA4BJUD
|
|||||||
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
github.com/go-toolsmith/strparse v1.0.0/go.mod h1:YI2nUKP9YGZnL/L1/DLFBfixrcjslWct4wyljWhSRy8=
|
||||||
github.com/go-toolsmith/typep v1.0.0 h1:zKymWyA1TRYvqYrYDrfEMZULyrhcnGY3x7LDKU2XQaA=
|
github.com/go-toolsmith/typep v1.0.0 h1:zKymWyA1TRYvqYrYDrfEMZULyrhcnGY3x7LDKU2XQaA=
|
||||||
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
github.com/go-toolsmith/typep v1.0.0/go.mod h1:JSQCQMUPdRlMZFswiq3TGpNp1GMktqkR2Ns5AIQkATU=
|
||||||
github.com/gobuffalo/here v0.6.0/go.mod h1:wAG085dHOYqUpf+Ap+WOdrPTp5IYcDAs/x7PLa8Y5fM=
|
|
||||||
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
|
github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
|
||||||
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
|
||||||
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b h1:ekuhfTjngPhisSjOJ0QWKpPQE8/rbknHaes6WVJj5Hw=
|
github.com/gofrs/flock v0.0.0-20190320160742-5135e617513b h1:ekuhfTjngPhisSjOJ0QWKpPQE8/rbknHaes6WVJj5Hw=
|
||||||
@@ -221,13 +164,11 @@ github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OI
|
|||||||
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
||||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4=
|
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaUGG7oYTSPP8MxqL4YI3kZKwcP4=
|
||||||
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
|
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ=
|
||||||
github.com/google/uuid v1.0.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
|
||||||
github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
||||||
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d h1:7XGaL1e6bYS1yIonGp9761ExpPPV1ui0SAC59Yube9k=
|
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d h1:7XGaL1e6bYS1yIonGp9761ExpPPV1ui0SAC59Yube9k=
|
||||||
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY=
|
github.com/googleapis/gnostic v0.0.0-20170729233727-0c5108395e2d/go.mod h1:sJBsCZ4ayReDTBIg8b9dl28c5xFWyhBTVRp3pOg5EKY=
|
||||||
github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8=
|
github.com/gophercloud/gophercloud v0.1.0/go.mod h1:vxM41WHh5uqHVBMZHzuwNOHh8XEoIEcSTewFxm1c5g8=
|
||||||
github.com/gorilla/websocket v1.4.0 h1:WDFjx/TMzVgy9VdMMQi2K2Emtwi2QcUQsztZ/zLaH/Q=
|
|
||||||
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
|
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
|
||||||
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3 h1:JVnpOZS+qxli+rgVl98ILOXVNbW+kb5wcxeGx8ShUIw=
|
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3 h1:JVnpOZS+qxli+rgVl98ILOXVNbW+kb5wcxeGx8ShUIw=
|
||||||
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3/go.mod h1:eEOZF4jCKGi+aprrirO9e7WKB3beBRtWgqGunKl6pKE=
|
github.com/gostaticanalysis/analysisutil v0.0.0-20190318220348-4088753ea4d3/go.mod h1:eEOZF4jCKGi+aprrirO9e7WKB3beBRtWgqGunKl6pKE=
|
||||||
@@ -235,24 +176,12 @@ github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:Fecb
|
|||||||
github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
|
github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
|
||||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
|
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
|
github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
|
||||||
github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA=
|
|
||||||
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
|
|
||||||
github.com/hashicorp/go-cleanhttp v0.5.0 h1:wvCrVc9TjDls6+YGAF2hAifE1E5U1+b4tH6KdvN3Gig=
|
|
||||||
github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80=
|
|
||||||
github.com/hashicorp/go-multierror v1.1.0 h1:B9UzwGQJehnUY1yNrnwREHc3fGbC2xefo8g4TbElacI=
|
|
||||||
github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA=
|
|
||||||
github.com/hashicorp/go-safetemp v1.0.0 h1:2HR189eFNrjHQyENnQMMpCiBAsRxzbTMIgBhEyExpmo=
|
|
||||||
github.com/hashicorp/go-safetemp v1.0.0/go.mod h1:oaerMy3BhqiTbVye6QuFhFtIceqFoDHxNAB65b+Rj1I=
|
|
||||||
github.com/hashicorp/go-version v1.1.0 h1:bPIoEKD27tNdebFGGxxYwcL4nepeY4j1QP23PFRGzg0=
|
|
||||||
github.com/hashicorp/go-version v1.1.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
|
|
||||||
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
github.com/hashicorp/golang-lru v0.5.1 h1:0hERBMJE1eitiLkihrMvRVBYAkpHzc/J3QdDN+dAcgU=
|
|
||||||
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
||||||
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
||||||
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
github.com/hpcloud/tail v1.0.0 h1:nfCOvKYfkgYP8hkirhJocXT2+zOD8yUNjXaWfTlyFKI=
|
||||||
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
||||||
github.com/imdario/mergo v0.3.5 h1:JboBksRwiiAJWvIYJVo46AfV+IAIKZpfrSzVKj42R4Q=
|
|
||||||
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
||||||
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
|
github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM=
|
||||||
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
|
||||||
@@ -285,13 +214,9 @@ github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e/go.mod h1:7rIyQ
|
|||||||
github.com/magiconair/properties v1.8.0 h1:LLgXmsheXeRoUOBOjtwPQCWIYqM/LU1ayDtDePerRcY=
|
github.com/magiconair/properties v1.8.0 h1:LLgXmsheXeRoUOBOjtwPQCWIYqM/LU1ayDtDePerRcY=
|
||||||
github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
|
github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ=
|
||||||
github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
||||||
github.com/mailru/easyjson v0.0.0-20180823135443-60711f1a8329/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
|
||||||
github.com/mailru/easyjson v0.0.0-20190312143242-1de009706dbe/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
|
||||||
github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
||||||
|
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e h1:hB2xlXdHp/pmPZq0y3QnmWAArdw9PqbmotexnWx/FU8=
|
||||||
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
||||||
github.com/mailru/easyjson v0.7.0 h1:aizVhC/NAAcKWb+5QsU1iNOZb4Yws5UO2I+aIprQITM=
|
|
||||||
github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs=
|
|
||||||
github.com/markbates/pkger v0.17.1/go.mod h1:0JoVlrol20BSywW79rN3kdFFsE5xYM+rSCQDXbLhiuI=
|
|
||||||
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb h1:RHba4YImhrUVQDHUCe2BNSOz4tVy2yGyXhvYDvxGgeE=
|
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb h1:RHba4YImhrUVQDHUCe2BNSOz4tVy2yGyXhvYDvxGgeE=
|
||||||
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
github.com/matoous/godox v0.0.0-20190911065817-5d6d842e92eb/go.mod h1:1BELzlh859Sh1c6+90blK8lbYy0kwQf1bYlBhBysy1s=
|
||||||
github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
|
github.com/mattn/go-colorable v0.1.4 h1:snbPLB8fVfU9iwbbo30TPtbLRzwWu6aJS6Xh4eaaviA=
|
||||||
@@ -300,12 +225,9 @@ github.com/mattn/go-isatty v0.0.8 h1:HLtExJ+uU2HOZ+wI0Tt5DtUDrx8yhUqDcp7fYERX4CE
|
|||||||
github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
|
||||||
github.com/mattn/goveralls v0.0.2/go.mod h1:8d1ZMHsd7fW6IRPKQh46F2WRpyib5/X4FOpevwGNQEw=
|
github.com/mattn/goveralls v0.0.2/go.mod h1:8d1ZMHsd7fW6IRPKQh46F2WRpyib5/X4FOpevwGNQEw=
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
||||||
github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
|
|
||||||
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
|
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
|
||||||
github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
|
github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
|
||||||
github.com/mitchellh/go-ps v0.0.0-20190716172923-621e5597135b/go.mod h1:r1VsdOzOPt1ZSrGZWFoNhsAedKnEd6r9Np1+5blZCWk=
|
github.com/mitchellh/go-ps v0.0.0-20190716172923-621e5597135b/go.mod h1:r1VsdOzOPt1ZSrGZWFoNhsAedKnEd6r9Np1+5blZCWk=
|
||||||
github.com/mitchellh/go-testing-interface v1.0.0 h1:fzU/JVNcaqHQEcVFAKeR41fkiLdIPrefOvVG1VZ96U0=
|
|
||||||
github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI=
|
|
||||||
github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE=
|
github.com/mitchellh/mapstructure v1.1.2 h1:fmNYVwqnSfB9mZU6OS2O6GsXM+wcskZDuKQzvN1EDeE=
|
||||||
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y=
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
@@ -315,9 +237,6 @@ github.com/modern-go/reflect2 v0.0.0-20180320133207-05fbef0ca5da/go.mod h1:bx2lN
|
|||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
|
github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
|
||||||
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
||||||
github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826/go.mod h1:TaXosZuwdSHYgviHp1DAtfrULt5eUgsSMsZf+YrPgl8=
|
|
||||||
github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 h1:n6/2gBQ3RWajuToeY6ZtZTIKv2v7ThUy5KKusIT0yc0=
|
|
||||||
github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00/go.mod h1:Pm3mSP3c5uWn86xMLZ5Sa7JB9GsEZySvHYXCTK4E9q4=
|
|
||||||
github.com/mozilla/tls-observatory v0.0.0-20190404164649-a3c1b6cfecfd/go.mod h1:SrKMQvPiws7F7iqYp8/TX+IhxCYhzr6N/1yb8cwHsGk=
|
github.com/mozilla/tls-observatory v0.0.0-20190404164649-a3c1b6cfecfd/go.mod h1:SrKMQvPiws7F7iqYp8/TX+IhxCYhzr6N/1yb8cwHsGk=
|
||||||
github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
|
github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
|
||||||
github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
|
github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
|
||||||
@@ -332,8 +251,6 @@ github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+
|
|||||||
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
||||||
github.com/onsi/gomega v1.7.0 h1:XPnZz8VVBHjVsy1vzJmRwIcSwiUO+JFfrv/xGiigmME=
|
github.com/onsi/gomega v1.7.0 h1:XPnZz8VVBHjVsy1vzJmRwIcSwiUO+JFfrv/xGiigmME=
|
||||||
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
||||||
github.com/paulmach/orb v0.1.3/go.mod h1:VFlX/8C+IQ1p6FTRRKzKoOPJnvEtA5G0Veuqwbu//Vk=
|
|
||||||
github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k=
|
|
||||||
github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc=
|
github.com/pelletier/go-toml v1.2.0 h1:T5zMGML61Wp+FlcbWjRDT7yAxhJNAiPPLOFECq181zc=
|
||||||
github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
|
github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
|
||||||
github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU=
|
github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU=
|
||||||
@@ -352,26 +269,18 @@ github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y8
|
|||||||
github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
|
github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
|
||||||
github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
|
github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
|
||||||
github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
|
github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
|
||||||
github.com/qri-io/starlib v0.4.2-0.20200213133954-ff2e8cd5ef8d h1:K6eOUihrFLdZjZnA4XlRp864fmWXv9YTIk7VPLhRacA=
|
|
||||||
github.com/qri-io/starlib v0.4.2-0.20200213133954-ff2e8cd5ef8d/go.mod h1:7DPO4domFU579Ga6E61sB9VFNaniPVwJP5C4bBCu3wA=
|
|
||||||
github.com/quasilyte/go-consistent v0.0.0-20190521200055-c6f3937de18c/go.mod h1:5STLWrekHfjyYwxBRVRXNOSewLJ3PWfDJd1VyTS21fI=
|
github.com/quasilyte/go-consistent v0.0.0-20190521200055-c6f3937de18c/go.mod h1:5STLWrekHfjyYwxBRVRXNOSewLJ3PWfDJd1VyTS21fI=
|
||||||
github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
|
github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
|
||||||
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
||||||
github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g=
|
||||||
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
|
||||||
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d h1:BzRvVq1EHuIjxpijCEKpAxzKUUMurOQ4sknehIATRh8=
|
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d h1:BzRvVq1EHuIjxpijCEKpAxzKUUMurOQ4sknehIATRh8=
|
||||||
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d/go.mod h1:w5+eXa0mYznDkHaMCXA4XYffjlH+cy1oyKbfzJXa2Do=
|
github.com/securego/gosec v0.0.0-20191002120514-e680875ea14d/go.mod h1:w5+eXa0mYznDkHaMCXA4XYffjlH+cy1oyKbfzJXa2Do=
|
||||||
github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo=
|
|
||||||
github.com/sergi/go-diff v1.1.0 h1:we8PVUC3FE2uYfodKH/nBHMSetSfHDR6scGdBi+erh0=
|
|
||||||
github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
|
|
||||||
github.com/shirou/gopsutil v0.0.0-20190901111213-e4ec7b275ada/go.mod h1:WWnYX4lzhCH5h/3YBfyVA3VbLYjlMZZAQcW9ojMexNc=
|
github.com/shirou/gopsutil v0.0.0-20190901111213-e4ec7b275ada/go.mod h1:WWnYX4lzhCH5h/3YBfyVA3VbLYjlMZZAQcW9ojMexNc=
|
||||||
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
|
github.com/shirou/w32 v0.0.0-20160930032740-bb4de0191aa4/go.mod h1:qsXQc7+bwAM3Q1u/4XEfrquwF8Lw7D7y5cD8CuHnfIc=
|
||||||
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e h1:MZM7FHLqUHYI0Y/mQAt3d2aYa0SiNms/hFqC9qJYolM=
|
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e h1:MZM7FHLqUHYI0Y/mQAt3d2aYa0SiNms/hFqC9qJYolM=
|
||||||
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk=
|
github.com/shurcooL/go v0.0.0-20180423040247-9e1955d9fb6e/go.mod h1:TDJrrUr11Vxrven61rcy3hJMUqaf/CLWYhHNPmT14Lk=
|
||||||
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041 h1:llrF3Fs4018ePo4+G/HV/uQUqEI1HMDjCeOf2V6puPc=
|
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041 h1:llrF3Fs4018ePo4+G/HV/uQUqEI1HMDjCeOf2V6puPc=
|
||||||
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ=
|
github.com/shurcooL/go-goon v0.0.0-20170922171312-37c2f522c041/go.mod h1:N5mDOmsrJOB+vfqUK+7DmDyjhSLIIBnXo9lvZJj3MWQ=
|
||||||
github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo=
|
|
||||||
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
|
|
||||||
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
||||||
github.com/sirupsen/logrus v1.4.2 h1:SPIRibHv4MatM3XXNO2BJeFLZwZ2LvZgfQ5+UNI2im4=
|
github.com/sirupsen/logrus v1.4.2 h1:SPIRibHv4MatM3XXNO2BJeFLZwZ2LvZgfQ5+UNI2im4=
|
||||||
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
||||||
@@ -384,9 +293,8 @@ github.com/spf13/afero v1.2.2 h1:5jhuqJyZCZf2JRofRvN/nIFgIWNzPa3/Vz8mYylgbWc=
|
|||||||
github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk=
|
github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk=
|
||||||
github.com/spf13/cast v1.3.0 h1:oget//CVOEoFewqQxwr0Ej5yjygnqGkvggSE/gB35Q8=
|
github.com/spf13/cast v1.3.0 h1:oget//CVOEoFewqQxwr0Ej5yjygnqGkvggSE/gB35Q8=
|
||||||
github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
|
github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
|
||||||
|
github.com/spf13/cobra v0.0.5 h1:f0B+LkLX6DtmRH1isoNA9VTtNUK9K8xYd28JNNfOv/s=
|
||||||
github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU=
|
github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU=
|
||||||
github.com/spf13/cobra v1.0.0 h1:6m/oheQuQ13N9ks4hubMG6BnvwOeaJrqSPLahSnczz8=
|
|
||||||
github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE=
|
|
||||||
github.com/spf13/jwalterweatherman v1.0.0 h1:XHEdyB+EcvlqZamSM4ZOMGlc93t6AcsBEu9Gc1vn7yk=
|
github.com/spf13/jwalterweatherman v1.0.0 h1:XHEdyB+EcvlqZamSM4ZOMGlc93t6AcsBEu9Gc1vn7yk=
|
||||||
github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
|
github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo=
|
||||||
github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
|
github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
|
||||||
@@ -402,18 +310,14 @@ github.com/stretchr/objx v0.2.0 h1:Hbg2NidpLE8veEBkEZTL3CvlkUIVzuU9jDplZO54c48=
|
|||||||
github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE=
|
github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE=
|
||||||
github.com/stretchr/testify v0.0.0-20151208002404-e3a8ff8ce365/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
github.com/stretchr/testify v0.0.0-20151208002404-e3a8ff8ce365/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||||
github.com/stretchr/testify v1.2.3-0.20181224173747-660f15d67dbb/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
|
github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
||||||
github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
|
|
||||||
github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e h1:RumXZ56IrCj4CL+g1b9OL/oH0QnsF976bC8xQFYUD5Q=
|
github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e h1:RumXZ56IrCj4CL+g1b9OL/oH0QnsF976bC8xQFYUD5Q=
|
||||||
github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e/go.mod h1:Qimiffbc6q9tBWlVV6x0P9sat/ao1xEkREYPPj9hphk=
|
github.com/timakin/bodyclose v0.0.0-20190930140734-f7f2e9bca95e/go.mod h1:Qimiffbc6q9tBWlVV6x0P9sat/ao1xEkREYPPj9hphk=
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
||||||
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
|
||||||
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0=
|
||||||
github.com/ulikunitz/xz v0.5.8 h1:ERv8V6GKqVi23rgu5cj9pVfVzJbOqAY2Ntl88O6c2nQ=
|
|
||||||
github.com/ulikunitz/xz v0.5.8/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
|
|
||||||
github.com/ultraware/funlen v0.0.2 h1:Av96YVBwwNSe4MLR7iI/BIa3VyI7/djnto/pK3Uxbdo=
|
github.com/ultraware/funlen v0.0.2 h1:Av96YVBwwNSe4MLR7iI/BIa3VyI7/djnto/pK3Uxbdo=
|
||||||
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
github.com/ultraware/funlen v0.0.2/go.mod h1:Dp4UiAus7Wdb9KUZsYWZEWiRzGuM2kXM1lPbfaF6xhA=
|
||||||
github.com/ultraware/whitespace v0.0.4 h1:If7Va4cM03mpgrNH9k49/VOicWpGoG70XPBFFODYDsg=
|
github.com/ultraware/whitespace v0.0.4 h1:If7Va4cM03mpgrNH9k49/VOicWpGoG70XPBFFODYDsg=
|
||||||
@@ -424,21 +328,10 @@ github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyC
|
|||||||
github.com/valyala/fasthttp v1.2.0/go.mod h1:4vX61m6KN+xDduDNwXrhIAVZaZaZiQ1luJk8LWSxF3s=
|
github.com/valyala/fasthttp v1.2.0/go.mod h1:4vX61m6KN+xDduDNwXrhIAVZaZaZiQ1luJk8LWSxF3s=
|
||||||
github.com/valyala/quicktemplate v1.2.0/go.mod h1:EH+4AkTd43SvgIbQHYu59/cJyxDoOVRUAfrukLPuGJ4=
|
github.com/valyala/quicktemplate v1.2.0/go.mod h1:EH+4AkTd43SvgIbQHYu59/cJyxDoOVRUAfrukLPuGJ4=
|
||||||
github.com/valyala/tcplisten v0.0.0-20161114210144-ceec8f93295a/go.mod h1:v3UYOV9WzVtRmSR+PDvWpU/qWl4Wa5LApYYX4ZtKbio=
|
github.com/valyala/tcplisten v0.0.0-20161114210144-ceec8f93295a/go.mod h1:v3UYOV9WzVtRmSR+PDvWpU/qWl4Wa5LApYYX4ZtKbio=
|
||||||
github.com/vektah/gqlparser v1.1.2/go.mod h1:1ycwN7Ij5njmMkPPAOaRFY4rET2Enx7IkVv3vaXspKw=
|
|
||||||
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
|
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
|
||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca h1:1CFlNzQhALwjS9mBAUkycX616GzgsuYUOCHA5+HSlXI=
|
|
||||||
github.com/xlab/treeprint v0.0.0-20181112141820-a009c3971eca/go.mod h1:ce1O1j6UtZfjr22oyGxGLbauSBp2YVXpARAosm7dHBg=
|
|
||||||
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf h1:gvEmqF83GB8R5XtrMseJb6A6R0OCtNAS8f4TmZg2dGc=
|
|
||||||
github.com/yujunz/go-getter v1.5.1-lite.0.20201201013212-6d9c071adddf/go.mod h1:bL0Pr07HEdsMZ1WBqZIxXj96r5LnFsY4LgPaPEGkw1k=
|
|
||||||
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
|
||||||
go.mongodb.org/mongo-driver v1.0.3/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
|
||||||
go.mongodb.org/mongo-driver v1.1.1/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
|
||||||
go.mongodb.org/mongo-driver v1.1.2/go.mod h1:u7ryQJ+DOzQmeO7zB6MHyr8jkEQvC8vH7qLUO4lqsUM=
|
|
||||||
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
||||||
go.starlark.net v0.0.0-20190528202925-30ae18b8564f/go.mod h1:c1/X6cHgvdXj6pUlmWKMkuqRnW4K8x2vwt6JAaaircg=
|
|
||||||
go.starlark.net v0.0.0-20200306205701-8dd3e2ee1dd5 h1:+FNtrFTmVw0YZGpBGX56XDee331t6JAXeK2bcyhLOOc=
|
|
||||||
go.starlark.net v0.0.0-20200306205701-8dd3e2ee1dd5/go.mod h1:nmDLcffg48OtT/PSW0Hg7FvpRQsQh5OSqIylirxKC7o=
|
|
||||||
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
||||||
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
||||||
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
|
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
|
||||||
@@ -446,12 +339,9 @@ golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnf
|
|||||||
golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
golang.org/x/crypto v0.0.0-20190211182817-74369b46fc67/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
golang.org/x/crypto v0.0.0-20190320223903-b7391e95e576/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
golang.org/x/crypto v0.0.0-20190617133340-57b3e21c3d56/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
|
||||||
golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392 h1:ACG4HJsFiNMf47Y4PeRoebLNy/2lXT9EtprMuTFWt1M=
|
|
||||||
golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY=
|
golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY=
|
||||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||||
@@ -460,18 +350,15 @@ golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTk
|
|||||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||||
golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
|
golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
|
||||||
golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20170114055629-f2499483f923/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180218175443-cbe0f9307d01/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20180911220305-26e67e76b6c3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20180911220305-26e67e76b6c3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20181005035420-146acd28ed58/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20190320064053-1272bf9dcd53/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
golang.org/x/net v0.0.0-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
|
golang.org/x/net v0.0.0-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
|
||||||
golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
@@ -482,7 +369,6 @@ golang.org/x/net v0.0.0-20191004110552-13f9640d40b9 h1:rjwSpXsdiK0dV8/Naq3kAw9ym
|
|||||||
golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||||
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45 h1:SVwTIAaPC2U/AvvLNZ2a7OVsmBpC8L5BlwK1whH3hm0=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
@@ -499,21 +385,18 @@ golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5h
|
|||||||
golang.org/x/sys v0.0.0-20190209173611-3b5209105503/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20190209173611-3b5209105503/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190321052220-f7bb7a8bee54/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20190922100055-0a153f010e69 h1:rOhMmluY6kLMhdnrivzec6lLgaVbMHMn2ISQXJeJ5EM=
|
||||||
golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
golang.org/x/sys v0.0.0-20191002063906-3421d5a6bb1c h1:Vco5b+cuG5NNfORVxZy6bYZQ7rsigisU1WQFkvQ0L5E=
|
|
||||||
golang.org/x/sys v0.0.0-20191002063906-3421d5a6bb1c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/text v0.0.0-20160726164857-2910a502d2bf/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.0.0-20160726164857-2910a502d2bf/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs=
|
golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs=
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4 h1:SvFZT6jyqRaOeXpc5h/JSfZenJ2O330aBsf7JfSUXmQ=
|
|
||||||
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
@@ -523,23 +406,20 @@ golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGm
|
|||||||
golang.org/x/tools v0.0.0-20181117154741-2ddaf7f79a09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20181117154741-2ddaf7f79a09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190110163146-51295c7ec13a/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20190110163146-51295c7ec13a/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190125232054-d66bd3c5d5a6/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
||||||
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||||
golang.org/x/tools v0.0.0-20190311215038-5c2858a9cfe5/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
golang.org/x/tools v0.0.0-20190311215038-5c2858a9cfe5/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||||
golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||||
golang.org/x/tools v0.0.0-20190322203728-c1a832b0ad89/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
golang.org/x/tools v0.0.0-20190322203728-c1a832b0ad89/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||||
golang.org/x/tools v0.0.0-20190521203540-521d6ed310dd/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
golang.org/x/tools v0.0.0-20190521203540-521d6ed310dd/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||||
|
golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59 h1:QjA/9ArTfVTLfEhClDCG7SGrZkZixxWpwNCDiwJfh88=
|
||||||
golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
||||||
golang.org/x/tools v0.0.0-20190617190820-da514acc4774/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
|
||||||
golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
||||||
golang.org/x/tools v0.0.0-20190719005602-e377ae9d6386/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI=
|
golang.org/x/tools v0.0.0-20190719005602-e377ae9d6386/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI=
|
||||||
golang.org/x/tools v0.0.0-20190910044552-dd2b5c81c578/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
golang.org/x/tools v0.0.0-20190910044552-dd2b5c81c578/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
golang.org/x/tools v0.0.0-20190930201159-7c411dea38b0/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
golang.org/x/tools v0.0.0-20190930201159-7c411dea38b0/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
golang.org/x/tools v0.0.0-20191010075000-0337d82405ff h1:XdBG6es/oFDr1HwaxkxgVve7NB281QhxgK/i4voubFs=
|
golang.org/x/tools v0.0.0-20191010075000-0337d82405ff h1:XdBG6es/oFDr1HwaxkxgVve7NB281QhxgK/i4voubFs=
|
||||||
golang.org/x/tools v0.0.0-20191010075000-0337d82405ff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
golang.org/x/tools v0.0.0-20191010075000-0337d82405ff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e h1:aZzprAO9/8oim3qStq3wc1Xuxx4QmAGriC4VU4ojemQ=
|
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
|
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
|
||||||
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
||||||
@@ -566,14 +446,8 @@ gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWD
|
|||||||
gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
|
gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
|
||||||
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
|
gopkg.in/yaml.v2 v2.2.4 h1:/eiJrUcujPVeJ3xlSWaiNi3uSVmDGBK1pDHUHAnao1I=
|
||||||
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||||
gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
|
|
||||||
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo=
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||||
honnef.co/go/tools v0.0.1-2019.2.3 h1:3JgtbtFHMiCmsznwGVTUWbgGov+pVqnlf1dEJTNAXeM=
|
honnef.co/go/tools v0.0.1-2019.2.3 h1:3JgtbtFHMiCmsznwGVTUWbgGov+pVqnlf1dEJTNAXeM=
|
||||||
@@ -591,7 +465,6 @@ k8s.io/klog v1.0.0 h1:Pt+yjF5aB1xDSVbau4VsWe+dQNzA0qv1LlXdC2dF6Q8=
|
|||||||
k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
|
k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
|
||||||
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a h1:UcxjrRMyNx/i/y8G7kPvLyy7rfbeuf1PYyBf973pgyU=
|
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a h1:UcxjrRMyNx/i/y8G7kPvLyy7rfbeuf1PYyBf973pgyU=
|
||||||
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a/go.mod h1:1TqjTSzOxsLGIKfj0lK8EeCP7K1iUG65v09OM0/WG5E=
|
k8s.io/kube-openapi v0.0.0-20191107075043-30be4d16710a/go.mod h1:1TqjTSzOxsLGIKfj0lK8EeCP7K1iUG65v09OM0/WG5E=
|
||||||
k8s.io/utils v0.0.0-20191114184206-e782cd3c129f h1:GiPwtSzdP43eI1hpPCbROQCCIgCuiMMNF8YUVLF3vJo=
|
|
||||||
k8s.io/utils v0.0.0-20191114184206-e782cd3c129f/go.mod h1:sZAwmy6armz5eXlNoLmJcl4F1QuKu7sr+mFQ0byX7Ew=
|
k8s.io/utils v0.0.0-20191114184206-e782cd3c129f/go.mod h1:sZAwmy6armz5eXlNoLmJcl4F1QuKu7sr+mFQ0byX7Ew=
|
||||||
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed h1:WX1yoOaKQfddO/mLzdV4wptyWgoH/6hwLs7QHTixo0I=
|
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed h1:WX1yoOaKQfddO/mLzdV4wptyWgoH/6hwLs7QHTixo0I=
|
||||||
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc=
|
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc=
|
||||||
@@ -599,11 +472,8 @@ mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b h1:DxJ5nJdkhDlLok9K6qO+5290kphD
|
|||||||
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4=
|
||||||
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f h1:Cq7MalBHYACRd6EesksG1Q8EoIAKOsiZviGKbOLIej4=
|
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f h1:Cq7MalBHYACRd6EesksG1Q8EoIAKOsiZviGKbOLIej4=
|
||||||
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
mvdan.cc/unparam v0.0.0-20190720180237-d51796306d8f/go.mod h1:4G1h5nDURzA3bwVMZIVpwbkw+04kSxk3rAtzlimaUJw=
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.9 h1:n3WNdvPPReRNDxW+XXd2JlyZ8EII721I21D1DBpBVBE=
|
|
||||||
sigs.k8s.io/kustomize/kyaml v0.10.9/go.mod h1:K9yg1k/HB/6xNOf5VH3LhTo1DK9/5ykSZO5uIv+Y/1k=
|
|
||||||
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
sigs.k8s.io/structured-merge-diff v0.0.0-20190525122527-15d366b2352e/go.mod h1:wWxsB5ozmmv/SG7nM11ayaAW51xMvak/t1r0CSlcokI=
|
||||||
|
sigs.k8s.io/yaml v1.1.0 h1:4A07+ZFc2wgJwo8YNlQpr1rVlgUDlxXHhPJciaPY5gs=
|
||||||
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
||||||
sigs.k8s.io/yaml v1.2.0 h1:kr/MCeFWJWTwyaHoR9c8EjH9OumOmoF9YGiZd7lFm/Q=
|
|
||||||
sigs.k8s.io/yaml v1.2.0/go.mod h1:yfXDCHCao9+ENCvLSE62v9VSji2MKu5jeNfTrofGhJc=
|
|
||||||
sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4 h1:JPJh2pk3+X4lXAkZIk2RuE/7/FoK9maXw+TNPJhVS/c=
|
sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4 h1:JPJh2pk3+X4lXAkZIk2RuE/7/FoK9maXw+TNPJhVS/c=
|
||||||
sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4/go.mod h1:ketZ/q3QxT9HOBeFhu6RdvsftgpsbFHBF5Cas6cDKZ0=
|
sourcegraph.com/sqs/pbtypes v0.0.0-20180604144634-d3ebe8f20ae4/go.mod h1:ketZ/q3QxT9HOBeFhu6RdvsftgpsbFHBF5Cas6cDKZ0=
|
||||||
|
|||||||
@@ -8,8 +8,6 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// SortArrayAndComputeHash sorts a string array and
|
// SortArrayAndComputeHash sorts a string array and
|
||||||
@@ -52,105 +50,3 @@ func Encode(hex string) (string, error) {
|
|||||||
func Hash(data string) string {
|
func Hash(data string) string {
|
||||||
return fmt.Sprintf("%x", sha256.Sum256([]byte(data)))
|
return fmt.Sprintf("%x", sha256.Sum256([]byte(data)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// HashRNode returns the hash value of input RNode
|
|
||||||
func HashRNode(node *yaml.RNode) (string, error) {
|
|
||||||
// get node kind
|
|
||||||
kindNode, err := node.Pipe(yaml.FieldMatcher{Name: "kind"})
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
kind := kindNode.YNode().Value
|
|
||||||
|
|
||||||
// calculate hash for different kinds
|
|
||||||
encoded := ""
|
|
||||||
switch kind {
|
|
||||||
case "ConfigMap":
|
|
||||||
encoded, err = encodeConfigMap(node)
|
|
||||||
case "Secret":
|
|
||||||
encoded, err = encodeSecret(node)
|
|
||||||
default:
|
|
||||||
var encodedBytes []byte
|
|
||||||
encodedBytes, err = json.Marshal(node.YNode())
|
|
||||||
encoded = string(encodedBytes)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return Encode(Hash(encoded))
|
|
||||||
}
|
|
||||||
|
|
||||||
func getNodeValues(node *yaml.RNode, paths []string) (map[string]interface{}, error) {
|
|
||||||
values := make(map[string]interface{})
|
|
||||||
for _, p := range paths {
|
|
||||||
vn, err := node.Pipe(yaml.Lookup(p))
|
|
||||||
if err != nil {
|
|
||||||
return map[string]interface{}{}, err
|
|
||||||
}
|
|
||||||
if vn == nil {
|
|
||||||
values[p] = ""
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if vn.YNode().Kind != yaml.ScalarNode {
|
|
||||||
vs, err := vn.MarshalJSON()
|
|
||||||
if err != nil {
|
|
||||||
return map[string]interface{}{}, err
|
|
||||||
}
|
|
||||||
// data, binaryData and stringData are all maps
|
|
||||||
var v map[string]interface{}
|
|
||||||
json.Unmarshal(vs, &v)
|
|
||||||
values[p] = v
|
|
||||||
} else {
|
|
||||||
values[p] = vn.YNode().Value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return values, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// encodeConfigMap encodes a ConfigMap.
|
|
||||||
// Data, Kind, and Name are taken into account.
|
|
||||||
// BinaryData is included if it's not empty to avoid useless key in output.
|
|
||||||
func encodeConfigMap(node *yaml.RNode) (string, error) {
|
|
||||||
// get fields
|
|
||||||
paths := []string{"metadata/name", "data", "binaryData"}
|
|
||||||
values, err := getNodeValues(node, paths)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
m := map[string]interface{}{"kind": "ConfigMap", "name": values["metadata/name"],
|
|
||||||
"data": values["data"]}
|
|
||||||
if _, ok := values["binaryData"].(map[string]interface{}); ok {
|
|
||||||
m["binaryData"] = values["binaryData"]
|
|
||||||
}
|
|
||||||
|
|
||||||
// json.Marshal sorts the keys in a stable order in the encoding
|
|
||||||
data, err := json.Marshal(m)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return string(data), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// encodeSecret encodes a Secret.
|
|
||||||
// Data, Kind, Name, and Type are taken into account.
|
|
||||||
// StringData is included if it's not empty to avoid useless key in output.
|
|
||||||
func encodeSecret(node *yaml.RNode) (string, error) {
|
|
||||||
// get fields
|
|
||||||
paths := []string{"type", "metadata/name", "data", "stringData"}
|
|
||||||
values, err := getNodeValues(node, paths)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
m := map[string]interface{}{"kind": "Secret", "type": values["type"],
|
|
||||||
"name": values["metadata/name"], "data": values["data"]}
|
|
||||||
if _, ok := values["stringData"].(map[string]interface{}); ok {
|
|
||||||
m["stringData"] = values["stringData"]
|
|
||||||
}
|
|
||||||
|
|
||||||
// json.Marshal sorts the keys in a stable order in the encoding
|
|
||||||
data, err := json.Marshal(m)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return string(data), nil
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,13 +1,12 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
package hasher
|
package hasher_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/kyaml/yaml"
|
. "sigs.k8s.io/kustomize/api/hasher"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSortArrayAndComputeHash(t *testing.T) {
|
func TestSortArrayAndComputeHash(t *testing.T) {
|
||||||
@@ -40,314 +39,3 @@ func TestHash(t *testing.T) {
|
|||||||
t.Errorf("expected hash %q but got %q", expect, sum)
|
t.Errorf("expected hash %q but got %q", expect, sum)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestConfigMapHash(t *testing.T) {
|
|
||||||
cases := []struct {
|
|
||||||
desc string
|
|
||||||
cmYaml string
|
|
||||||
hash string
|
|
||||||
err string
|
|
||||||
}{
|
|
||||||
// empty map
|
|
||||||
{"empty data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap`, "6ct58987ht", ""},
|
|
||||||
// one key
|
|
||||||
{"one key", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
one: ""`, "9g67k2htb6", ""},
|
|
||||||
// three keys (tests sorting order)
|
|
||||||
{"three keys", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, "7757f9kkct", ""},
|
|
||||||
// empty binary data map
|
|
||||||
{"empty binary data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap`, "6ct58987ht", ""},
|
|
||||||
// one key with binary data
|
|
||||||
{"one key with binary data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
binaryData:
|
|
||||||
one: ""`, "6mtk2m274t", ""},
|
|
||||||
// three keys with binary data (tests sorting order)
|
|
||||||
{"three keys with binary data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
binaryData:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, "9th7kc28dg", ""},
|
|
||||||
// two keys, one with string and another with binary data
|
|
||||||
{"two keys with one each", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
one: ""
|
|
||||||
binaryData:
|
|
||||||
two: ""`, "698h7c7t9m", ""},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, c := range cases {
|
|
||||||
node, err := yaml.Parse(c.cmYaml)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
h, err := HashRNode(node)
|
|
||||||
if SkipRest(t, c.desc, err, c.err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if c.hash != h {
|
|
||||||
t.Errorf("case %q, expect hash %q but got %q", c.desc, c.hash, h)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSecretHash(t *testing.T) {
|
|
||||||
cases := []struct {
|
|
||||||
desc string
|
|
||||||
secretYaml string
|
|
||||||
hash string
|
|
||||||
err string
|
|
||||||
}{
|
|
||||||
// empty map
|
|
||||||
{"empty data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type`, "5gmgkf8578", ""},
|
|
||||||
// one key
|
|
||||||
{"one key", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""`, "74bd68bm66", ""},
|
|
||||||
// three keys (tests sorting order)
|
|
||||||
{"three keys", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, "4gf75c7476", ""},
|
|
||||||
// with stringdata
|
|
||||||
{"stringdata", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""
|
|
||||||
stringData:
|
|
||||||
two: 2`, "c4h4264gdb", ""},
|
|
||||||
// empty stringdata
|
|
||||||
{"empty stringdata", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""`, "74bd68bm66", ""},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, c := range cases {
|
|
||||||
node, err := yaml.Parse(c.secretYaml)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
h, err := HashRNode(node)
|
|
||||||
if SkipRest(t, c.desc, err, c.err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if c.hash != h {
|
|
||||||
t.Errorf("case %q, expect hash %q but got %q", c.desc, c.hash, h)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUnstructuredHash(t *testing.T) {
|
|
||||||
cases := []struct {
|
|
||||||
desc string
|
|
||||||
unstructured string
|
|
||||||
hash string
|
|
||||||
err string
|
|
||||||
}{
|
|
||||||
{"minimal", `
|
|
||||||
apiVersion: test/v1
|
|
||||||
kind: TestResource
|
|
||||||
metadata:
|
|
||||||
name: my-resource`, "244782mkb7", ""},
|
|
||||||
{"with spec", `
|
|
||||||
apiVersion: test/v1
|
|
||||||
kind: TestResource
|
|
||||||
metadata:
|
|
||||||
name: my-resource
|
|
||||||
spec:
|
|
||||||
foo: 1
|
|
||||||
bar: abc`, "59m2mdccg4", ""},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, c := range cases {
|
|
||||||
node, err := yaml.Parse(c.unstructured)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
h, err := HashRNode(node)
|
|
||||||
if SkipRest(t, c.desc, err, c.err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if c.hash != h {
|
|
||||||
t.Errorf("case %q, expect hash %q but got %q", c.desc, c.hash, h)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEncodeConfigMap(t *testing.T) {
|
|
||||||
cases := []struct {
|
|
||||||
desc string
|
|
||||||
cmYaml string
|
|
||||||
expect string
|
|
||||||
err string
|
|
||||||
}{
|
|
||||||
// empty map
|
|
||||||
{"empty data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap`, `{"data":"","kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// one key
|
|
||||||
{"one key", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
one: ""`, `{"data":{"one":""},"kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// three keys (tests sorting order)
|
|
||||||
{"three keys", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, `{"data":{"one":"","three":3,"two":2},"kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// empty binary map
|
|
||||||
{"empty data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap`, `{"data":"","kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// one key with binary data
|
|
||||||
{"one key", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
binaryData:
|
|
||||||
one: ""`, `{"binaryData":{"one":""},"data":"","kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// three keys with binary data (tests sorting order)
|
|
||||||
{"three keys", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
binaryData:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, `{"binaryData":{"one":"","three":3,"two":2},"data":"","kind":"ConfigMap","name":""}`, ""},
|
|
||||||
// two keys, one string and one binary values
|
|
||||||
{"two keys with one each", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
one: ""
|
|
||||||
binaryData:
|
|
||||||
two: ""`, `{"binaryData":{"two":""},"data":{"one":""},"kind":"ConfigMap","name":""}`, ""},
|
|
||||||
}
|
|
||||||
for _, c := range cases {
|
|
||||||
node, err := yaml.Parse(c.cmYaml)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
s, err := encodeConfigMap(node)
|
|
||||||
if SkipRest(t, c.desc, err, c.err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if s != c.expect {
|
|
||||||
t.Errorf("case %q, expect %q but got %q from encode %#v", c.desc, c.expect, s, c.cmYaml)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestEncodeSecret(t *testing.T) {
|
|
||||||
cases := []struct {
|
|
||||||
desc string
|
|
||||||
secretYaml string
|
|
||||||
expect string
|
|
||||||
err string
|
|
||||||
}{
|
|
||||||
// empty map
|
|
||||||
{"empty data", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type`, `{"data":"","kind":"Secret","name":"","type":"my-type"}`, ""},
|
|
||||||
// one key
|
|
||||||
{"one key", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""`, `{"data":{"one":""},"kind":"Secret","name":"","type":"my-type"}`, ""},
|
|
||||||
// three keys (tests sorting order) - note json.Marshal base64 encodes the values because they come in as []byte
|
|
||||||
{"three keys", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
two: 2
|
|
||||||
one: ""
|
|
||||||
three: 3`, `{"data":{"one":"","three":3,"two":2},"kind":"Secret","name":"","type":"my-type"}`, ""},
|
|
||||||
// with stringdata
|
|
||||||
{"stringdata", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""
|
|
||||||
stringData:
|
|
||||||
two: 2`, `{"data":{"one":""},"kind":"Secret","name":"","stringData":{"two":2},"type":"my-type"}`, ""},
|
|
||||||
// empty stringdata
|
|
||||||
{"empty stringdata", `
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Secret
|
|
||||||
type: my-type
|
|
||||||
data:
|
|
||||||
one: ""`, `{"data":{"one":""},"kind":"Secret","name":"","type":"my-type"}`, ""},
|
|
||||||
}
|
|
||||||
for _, c := range cases {
|
|
||||||
node, err := yaml.Parse(c.secretYaml)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
s, err := encodeSecret(node)
|
|
||||||
if SkipRest(t, c.desc, err, c.err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if s != c.expect {
|
|
||||||
t.Errorf("case %q, expect %q but got %q from encode %#v", c.desc, c.expect, s, c.secretYaml)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SkipRest returns true if there was a non-nil error or if we expected an error that didn't happen,
|
|
||||||
// and logs the appropriate error on the test object.
|
|
||||||
// The return value indicates whether we should skip the rest of the test case due to the error result.
|
|
||||||
func SkipRest(t *testing.T, desc string, err error, contains string) bool {
|
|
||||||
if err != nil {
|
|
||||||
if len(contains) == 0 {
|
|
||||||
t.Errorf("case %q, expect nil error but got %q", desc, err.Error())
|
|
||||||
} else if !strings.Contains(err.Error(), contains) {
|
|
||||||
t.Errorf("case %q, expect error to contain %q but got %q", desc, contains, err.Error())
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
} else if len(contains) > 0 {
|
|
||||||
t.Errorf("case %q, expect error to contain %q but got nil error", desc, contains)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -38,72 +38,36 @@ type Loader interface {
|
|||||||
Cleanup() error
|
Cleanup() error
|
||||||
}
|
}
|
||||||
|
|
||||||
// Kunstructured represents a Kubernetes Resource Model object.
|
// Kunstructured allows manipulation of k8s objects
|
||||||
|
// that do not have Golang structs.
|
||||||
type Kunstructured interface {
|
type Kunstructured interface {
|
||||||
// Several uses.
|
|
||||||
Copy() Kunstructured
|
|
||||||
|
|
||||||
// GetAnnotations returns the k8s annotations.
|
|
||||||
GetAnnotations() map[string]string
|
|
||||||
|
|
||||||
// GetData returns a top-level "data" field, as in a ConfigMap.
|
|
||||||
GetDataMap() map[string]string
|
|
||||||
|
|
||||||
// Used by ResAccumulator and ReplacementTransformer.
|
|
||||||
GetFieldValue(string) (interface{}, error)
|
|
||||||
|
|
||||||
// Used by Resource.OrgId
|
|
||||||
GetGvk() resid.Gvk
|
|
||||||
|
|
||||||
// Used by resource.Factory.SliceFromBytes
|
|
||||||
GetKind() string
|
|
||||||
|
|
||||||
// GetLabels returns the k8s labels.
|
|
||||||
GetLabels() map[string]string
|
|
||||||
|
|
||||||
// Used by Resource.CurId and resource factory.
|
|
||||||
GetName() string
|
|
||||||
|
|
||||||
// Used by special case code in
|
|
||||||
// ResMap.SubsetThatCouldBeReferencedByResource
|
|
||||||
GetSlice(path string) ([]interface{}, error)
|
|
||||||
|
|
||||||
// GetString returns the value of a string field.
|
|
||||||
// Used by Resource.GetNamespace
|
|
||||||
GetString(string) (string, error)
|
|
||||||
|
|
||||||
// Several uses.
|
|
||||||
Map() map[string]interface{}
|
Map() map[string]interface{}
|
||||||
|
SetMap(map[string]interface{})
|
||||||
// Used by Resource.AsYAML and Resource.String
|
Copy() Kunstructured
|
||||||
|
GetFieldValue(string) (interface{}, error)
|
||||||
|
GetString(string) (string, error)
|
||||||
|
GetStringSlice(string) ([]string, error)
|
||||||
|
GetBool(path string) (bool, error)
|
||||||
|
GetFloat64(path string) (float64, error)
|
||||||
|
GetInt64(path string) (int64, error)
|
||||||
|
GetSlice(path string) ([]interface{}, error)
|
||||||
|
GetStringMap(path string) (map[string]string, error)
|
||||||
|
GetMap(path string) (map[string]interface{}, error)
|
||||||
MarshalJSON() ([]byte, error)
|
MarshalJSON() ([]byte, error)
|
||||||
|
|
||||||
// Used by resWrangler.Select
|
|
||||||
MatchesAnnotationSelector(selector string) (bool, error)
|
|
||||||
|
|
||||||
// Used by resWrangler.Select
|
|
||||||
MatchesLabelSelector(selector string) (bool, error)
|
|
||||||
|
|
||||||
// SetAnnotations replaces the k8s annotations.
|
|
||||||
SetAnnotations(map[string]string)
|
|
||||||
|
|
||||||
// SetDataMap sets a top-level "data" field, as in a ConfigMap.
|
|
||||||
SetDataMap(map[string]string)
|
|
||||||
|
|
||||||
// Used by PatchStrategicMergeTransformer.
|
|
||||||
SetGvk(resid.Gvk)
|
|
||||||
|
|
||||||
// SetLabels replaces the k8s labels.
|
|
||||||
SetLabels(map[string]string)
|
|
||||||
|
|
||||||
// SetName changes the name.
|
|
||||||
SetName(string)
|
|
||||||
|
|
||||||
// SetNamespace changes the namespace.
|
|
||||||
SetNamespace(string)
|
|
||||||
|
|
||||||
// Needed, for now, by kyaml/filtersutil.ApplyToJSON.
|
|
||||||
UnmarshalJSON([]byte) error
|
UnmarshalJSON([]byte) error
|
||||||
|
GetGvk() resid.Gvk
|
||||||
|
SetGvk(resid.Gvk)
|
||||||
|
GetKind() string
|
||||||
|
GetName() string
|
||||||
|
SetName(string)
|
||||||
|
SetNamespace(string)
|
||||||
|
GetLabels() map[string]string
|
||||||
|
SetLabels(map[string]string)
|
||||||
|
GetAnnotations() map[string]string
|
||||||
|
SetAnnotations(map[string]string)
|
||||||
|
MatchesLabelSelector(selector string) (bool, error)
|
||||||
|
MatchesAnnotationSelector(selector string) (bool, error)
|
||||||
|
Patch(Kunstructured) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// KunstructuredFactory makes instances of Kunstructured.
|
// KunstructuredFactory makes instances of Kunstructured.
|
||||||
@@ -111,8 +75,14 @@ type KunstructuredFactory interface {
|
|||||||
SliceFromBytes([]byte) ([]Kunstructured, error)
|
SliceFromBytes([]byte) ([]Kunstructured, error)
|
||||||
FromMap(m map[string]interface{}) Kunstructured
|
FromMap(m map[string]interface{}) Kunstructured
|
||||||
Hasher() KunstructuredHasher
|
Hasher() KunstructuredHasher
|
||||||
MakeConfigMap(kvLdr KvLoader, args *types.ConfigMapArgs) (Kunstructured, error)
|
MakeConfigMap(
|
||||||
MakeSecret(kvLdr KvLoader, args *types.SecretArgs) (Kunstructured, error)
|
kvLdr KvLoader,
|
||||||
|
options *types.GeneratorOptions,
|
||||||
|
args *types.ConfigMapArgs) (Kunstructured, error)
|
||||||
|
MakeSecret(
|
||||||
|
kvLdr KvLoader,
|
||||||
|
options *types.GeneratorOptions,
|
||||||
|
args *types.SecretArgs) (Kunstructured, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// KunstructuredHasher returns a hash of the argument
|
// KunstructuredHasher returns a hash of the argument
|
||||||
|
|||||||
@@ -1,50 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package image
|
|
||||||
|
|
||||||
import (
|
|
||||||
"regexp"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
// IsImageMatched returns true if the value of t is identical to the
|
|
||||||
// image name in the full image name and tag as given by s.
|
|
||||||
func IsImageMatched(s, t string) bool {
|
|
||||||
// Tag values are limited to [a-zA-Z0-9_.{}-].
|
|
||||||
// Some tools like Bazel rules_k8s allow tag patterns with {} characters.
|
|
||||||
// More info: https://github.com/bazelbuild/rules_k8s/pull/423
|
|
||||||
pattern, _ := regexp.Compile("^" + t + "(@sha256)?(:[a-zA-Z0-9_.{}-]*)?$")
|
|
||||||
return pattern.MatchString(s)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Split separates and returns the name and tag parts
|
|
||||||
// from the image string using either colon `:` or at `@` separators.
|
|
||||||
// Note that the returned tag keeps its separator.
|
|
||||||
func Split(imageName string) (name string, tag string) {
|
|
||||||
// check if image name contains a domain
|
|
||||||
// if domain is present, ignore domain and check for `:`
|
|
||||||
ic := -1
|
|
||||||
if slashIndex := strings.Index(imageName, "/"); slashIndex < 0 {
|
|
||||||
ic = strings.LastIndex(imageName, ":")
|
|
||||||
} else {
|
|
||||||
lastIc := strings.LastIndex(imageName[slashIndex:], ":")
|
|
||||||
// set ic only if `:` is present
|
|
||||||
if lastIc > 0 {
|
|
||||||
ic = slashIndex + lastIc
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ia := strings.LastIndex(imageName, "@")
|
|
||||||
if ic < 0 && ia < 0 {
|
|
||||||
return imageName, ""
|
|
||||||
}
|
|
||||||
|
|
||||||
i := ic
|
|
||||||
if ia > 0 {
|
|
||||||
i = ia
|
|
||||||
}
|
|
||||||
|
|
||||||
name = imageName[:i]
|
|
||||||
tag = imageName[i:]
|
|
||||||
return
|
|
||||||
}
|
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
// Copyright 2020 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package image
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestIsImageMatched(t *testing.T) {
|
|
||||||
testCases := []struct {
|
|
||||||
testName string
|
|
||||||
value string
|
|
||||||
name string
|
|
||||||
isMatched bool
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
testName: "identical",
|
|
||||||
value: "nginx",
|
|
||||||
name: "nginx",
|
|
||||||
isMatched: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "name is match",
|
|
||||||
value: "nginx:12345",
|
|
||||||
name: "nginx",
|
|
||||||
isMatched: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "name is not a match",
|
|
||||||
value: "apache:12345",
|
|
||||||
name: "nginx",
|
|
||||||
isMatched: false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tc := range testCases {
|
|
||||||
t.Run(tc.testName, func(t *testing.T) {
|
|
||||||
assert.Equal(t, tc.isMatched, IsImageMatched(tc.value, tc.name))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSplit(t *testing.T) {
|
|
||||||
testCases := []struct {
|
|
||||||
testName string
|
|
||||||
value string
|
|
||||||
name string
|
|
||||||
tag string
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
testName: "no tag",
|
|
||||||
value: "nginx",
|
|
||||||
name: "nginx",
|
|
||||||
tag: "",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "with tag",
|
|
||||||
value: "nginx:1.2.3",
|
|
||||||
name: "nginx",
|
|
||||||
tag: ":1.2.3",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
testName: "with digest",
|
|
||||||
value: "nginx@12345",
|
|
||||||
name: "nginx",
|
|
||||||
tag: "@12345",
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tc := range testCases {
|
|
||||||
t.Run(tc.testName, func(t *testing.T) {
|
|
||||||
name, tag := Split(tc.value)
|
|
||||||
assert.Equal(t, tc.name, name)
|
|
||||||
assert.Equal(t, tc.tag, tag)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
121
api/internal/accumulator/expansion/expand.go
Normal file
121
api/internal/accumulator/expansion/expand.go
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
// Package expansion provides functions find and replace $(FOO) style variables in strings.
|
||||||
|
package expansion
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
operator = '$'
|
||||||
|
referenceOpener = '('
|
||||||
|
referenceCloser = ')'
|
||||||
|
)
|
||||||
|
|
||||||
|
// syntaxWrap returns the input string wrapped by the expansion syntax.
|
||||||
|
func syntaxWrap(input string) string {
|
||||||
|
return string(operator) + string(referenceOpener) + input + string(referenceCloser)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MappingFuncFor returns a mapping function for use with Expand that
|
||||||
|
// implements the expansion semantics defined in the expansion spec; it
|
||||||
|
// returns the input string wrapped in the expansion syntax if no mapping
|
||||||
|
// for the input is found.
|
||||||
|
func MappingFuncFor(
|
||||||
|
counts map[string]int,
|
||||||
|
context ...map[string]interface{}) func(string) interface{} {
|
||||||
|
return func(input string) interface{} {
|
||||||
|
for _, vars := range context {
|
||||||
|
val, ok := vars[input]
|
||||||
|
if ok {
|
||||||
|
counts[input]++
|
||||||
|
switch typedV := val.(type) {
|
||||||
|
case string, int64, float64, bool:
|
||||||
|
return typedV
|
||||||
|
default:
|
||||||
|
return syntaxWrap(input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return syntaxWrap(input)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Expand replaces variable references in the input string according to
|
||||||
|
// the expansion spec using the given mapping function to resolve the
|
||||||
|
// values of variables.
|
||||||
|
func Expand(input string, mapping func(string) interface{}) interface{} {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
checkpoint := 0
|
||||||
|
for cursor := 0; cursor < len(input); cursor++ {
|
||||||
|
if input[cursor] == operator && cursor+1 < len(input) {
|
||||||
|
// Copy the portion of the input string since the last
|
||||||
|
// checkpoint into the buffer
|
||||||
|
buf.WriteString(input[checkpoint:cursor])
|
||||||
|
|
||||||
|
// Attempt to read the variable name as defined by the
|
||||||
|
// syntax from the input string
|
||||||
|
read, isVar, advance := tryReadVariableName(input[cursor+1:])
|
||||||
|
|
||||||
|
if isVar {
|
||||||
|
// We were able to read a variable name correctly;
|
||||||
|
// apply the mapping to the variable name and copy the
|
||||||
|
// bytes into the buffer
|
||||||
|
mapped := mapping(read)
|
||||||
|
if input == syntaxWrap(read) {
|
||||||
|
// Preserve the type of variable
|
||||||
|
return mapped
|
||||||
|
}
|
||||||
|
|
||||||
|
// Variable is used in a middle of a string
|
||||||
|
buf.WriteString(fmt.Sprintf("%v", mapped))
|
||||||
|
} else {
|
||||||
|
// Not a variable name; copy the read bytes into the buffer
|
||||||
|
buf.WriteString(read)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Advance the cursor in the input string to account for
|
||||||
|
// bytes consumed to read the variable name expression
|
||||||
|
cursor += advance
|
||||||
|
|
||||||
|
// Advance the checkpoint in the input string
|
||||||
|
checkpoint = cursor + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the buffer and any remaining unwritten bytes in the
|
||||||
|
// input string.
|
||||||
|
return buf.String() + input[checkpoint:]
|
||||||
|
}
|
||||||
|
|
||||||
|
// tryReadVariableName attempts to read a variable name from the input
|
||||||
|
// string and returns the content read from the input, whether that content
|
||||||
|
// represents a variable name to perform mapping on, and the number of bytes
|
||||||
|
// consumed in the input string.
|
||||||
|
//
|
||||||
|
// The input string is assumed not to contain the initial operator.
|
||||||
|
func tryReadVariableName(input string) (string, bool, int) {
|
||||||
|
switch input[0] {
|
||||||
|
case operator:
|
||||||
|
// Escaped operator; return it.
|
||||||
|
return input[0:1], false, 1
|
||||||
|
case referenceOpener:
|
||||||
|
// Scan to expression closer
|
||||||
|
for i := 1; i < len(input); i++ {
|
||||||
|
if input[i] == referenceCloser {
|
||||||
|
return input[1:i], true, i + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Incomplete reference; return it.
|
||||||
|
return string(operator) + string(referenceOpener), false, 1
|
||||||
|
default:
|
||||||
|
// Not the beginning of an expression, ie, an operator
|
||||||
|
// that doesn't begin an expression. Return the operator
|
||||||
|
// and the first rune in the string.
|
||||||
|
return string(operator) + string(input[0]), false, 1
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
// Copyright 2019 The Kubernetes Authors.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
package refvar_test
|
package expansion_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
. "sigs.k8s.io/kustomize/api/internal/accumulator/expansion"
|
||||||
. "sigs.k8s.io/kustomize/api/filters/refvar"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type expected struct {
|
type expected struct {
|
||||||
@@ -16,48 +15,6 @@ type expected struct {
|
|||||||
edited string
|
edited string
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPrimitiveReplacer(t *testing.T) {
|
|
||||||
varCounts := make(map[string]int)
|
|
||||||
f := MakePrimitiveReplacer(
|
|
||||||
varCounts,
|
|
||||||
map[string]interface{}{
|
|
||||||
"FOO": "bar",
|
|
||||||
"ZOO": "$(FOO)-1",
|
|
||||||
"BLU": "$(ZOO)-2",
|
|
||||||
"EIGHT": 8,
|
|
||||||
"PI": 3.14159,
|
|
||||||
"ZINT": "$(INT)",
|
|
||||||
"BOOL": "true",
|
|
||||||
"HUGENUMBER": int64(9223372036854775807),
|
|
||||||
"CRAZYMAP": map[string]int{"crazy": 200},
|
|
||||||
"ZBOOL": "$(BOOL)",
|
|
||||||
})
|
|
||||||
assert.Equal(t, "$()", f(""))
|
|
||||||
assert.Equal(t, "$( )", f(" "))
|
|
||||||
assert.Equal(t, "$(florida)", f("florida"))
|
|
||||||
assert.Equal(t, "$(0)", f("0"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, "bar", f("FOO"))
|
|
||||||
assert.Equal(t, 8, f("EIGHT"))
|
|
||||||
assert.Equal(t, 8, f("EIGHT"))
|
|
||||||
assert.Equal(t, 3.14159, f("PI"))
|
|
||||||
assert.Equal(t, "true", f("BOOL"))
|
|
||||||
assert.Equal(t, int64(9223372036854775807), f("HUGENUMBER"))
|
|
||||||
assert.Equal(t, "$(FOO)-1", f("ZOO"))
|
|
||||||
assert.Equal(t, "$(CRAZYMAP)", f("CRAZYMAP"))
|
|
||||||
assert.Equal(t,
|
|
||||||
map[string]int{
|
|
||||||
"FOO": 3,
|
|
||||||
"EIGHT": 2,
|
|
||||||
"BOOL": 1,
|
|
||||||
"PI": 1,
|
|
||||||
"ZOO": 1,
|
|
||||||
"HUGENUMBER": 1,
|
|
||||||
},
|
|
||||||
varCounts)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestMapReference(t *testing.T) {
|
func TestMapReference(t *testing.T) {
|
||||||
type env struct {
|
type env struct {
|
||||||
Name string
|
Name string
|
||||||
@@ -94,7 +51,7 @@ func TestMapReference(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
varMap := map[string]interface{}{
|
declaredEnv := map[string]interface{}{
|
||||||
"FOO": "bar",
|
"FOO": "bar",
|
||||||
"ZOO": "$(FOO)-1",
|
"ZOO": "$(FOO)-1",
|
||||||
"BLU": "$(ZOO)-2",
|
"BLU": "$(ZOO)-2",
|
||||||
@@ -104,11 +61,11 @@ func TestMapReference(t *testing.T) {
|
|||||||
"ZBOOL": "$(BOOL)",
|
"ZBOOL": "$(BOOL)",
|
||||||
}
|
}
|
||||||
|
|
||||||
varCounts := make(map[string]int)
|
counts := make(map[string]int)
|
||||||
|
mapping := MappingFuncFor(counts, declaredEnv)
|
||||||
|
|
||||||
for _, env := range envs {
|
for _, env := range envs {
|
||||||
varMap[env.Name] = DoReplacements(
|
declaredEnv[env.Name] = Expand(fmt.Sprintf("%v", env.Value), mapping)
|
||||||
fmt.Sprintf("%v", env.Value),
|
|
||||||
MakePrimitiveReplacer(varCounts, varMap))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
expectedEnv := map[string]expected{
|
expectedEnv := map[string]expected{
|
||||||
@@ -122,20 +79,45 @@ func TestMapReference(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for k, v := range expectedEnv {
|
for k, v := range expectedEnv {
|
||||||
if e, a := v, varMap[k]; e.edited != a || e.count != varCounts[k] {
|
if e, a := v, declaredEnv[k]; e.edited != a || e.count != counts[k] {
|
||||||
t.Errorf("Expected %v count=%d, got %v count=%d",
|
t.Errorf("Expected %v count=%d, got %v count=%d",
|
||||||
e.edited, e.count, a, varCounts[k])
|
e.edited, e.count, a, counts[k])
|
||||||
} else {
|
} else {
|
||||||
delete(varMap, k)
|
delete(declaredEnv, k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(varMap) != 0 {
|
if len(declaredEnv) != 0 {
|
||||||
t.Errorf("Unexpected keys in declared env: %v", varMap)
|
t.Errorf("Unexpected keys in declared env: %v", declaredEnv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestMapping(t *testing.T) {
|
func TestMapping(t *testing.T) {
|
||||||
|
context := map[string]interface{}{
|
||||||
|
"VAR_A": "A",
|
||||||
|
"VAR_B": "B",
|
||||||
|
"VAR_C": "C",
|
||||||
|
"VAR_REF": "$(VAR_A)",
|
||||||
|
"VAR_EMPTY": "",
|
||||||
|
}
|
||||||
|
doExpansionTest(t, context)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMappingDual(t *testing.T) {
|
||||||
|
context := map[string]interface{}{
|
||||||
|
"VAR_A": "A",
|
||||||
|
"VAR_EMPTY": "",
|
||||||
|
}
|
||||||
|
context2 := map[string]interface{}{
|
||||||
|
"VAR_B": "B",
|
||||||
|
"VAR_C": "C",
|
||||||
|
"VAR_REF": "$(VAR_A)",
|
||||||
|
}
|
||||||
|
|
||||||
|
doExpansionTest(t, context, context2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func doExpansionTest(t *testing.T, context ...map[string]interface{}) {
|
||||||
cases := []struct {
|
cases := []struct {
|
||||||
name string
|
name string
|
||||||
input string
|
input string
|
||||||
@@ -351,17 +333,11 @@ func TestMapping(t *testing.T) {
|
|||||||
expected: "\n",
|
expected: "\n",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range cases {
|
for _, tc := range cases {
|
||||||
counts := make(map[string]int)
|
counts := make(map[string]int)
|
||||||
expanded := DoReplacements(
|
mapping := MappingFuncFor(counts, context...)
|
||||||
fmt.Sprintf("%v", tc.input),
|
expanded := Expand(fmt.Sprintf("%v", tc.input), mapping)
|
||||||
MakePrimitiveReplacer(counts, map[string]interface{}{
|
|
||||||
"VAR_A": "A",
|
|
||||||
"VAR_B": "B",
|
|
||||||
"VAR_C": "C",
|
|
||||||
"VAR_REF": "$(VAR_A)",
|
|
||||||
"VAR_EMPTY": "",
|
|
||||||
}))
|
|
||||||
if e, a := tc.expected, expanded; e != a {
|
if e, a := tc.expected, expanded; e != a {
|
||||||
t.Errorf("%v: expected %q, got %q", tc.name, e, a)
|
t.Errorf("%v: expected %q, got %q", tc.name, e, a)
|
||||||
}
|
}
|
||||||
@@ -371,7 +347,8 @@ func TestMapping(t *testing.T) {
|
|||||||
}
|
}
|
||||||
if len(tc.counts) > 0 {
|
if len(tc.counts) > 0 {
|
||||||
for k, expectedCount := range tc.counts {
|
for k, expectedCount := range tc.counts {
|
||||||
if c, ok := counts[k]; ok {
|
c, ok := counts[k]
|
||||||
|
if ok {
|
||||||
if c != expectedCount {
|
if c != expectedCount {
|
||||||
t.Errorf(
|
t.Errorf(
|
||||||
"%v: k=%s, expected count %d, got %d",
|
"%v: k=%s, expected count %d, got %d",
|
||||||
@@ -9,6 +9,7 @@ import (
|
|||||||
|
|
||||||
"github.com/go-openapi/spec"
|
"github.com/go-openapi/spec"
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
"k8s.io/kube-openapi/pkg/common"
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/api/ifc"
|
"sigs.k8s.io/kustomize/api/ifc"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
@@ -17,16 +18,8 @@ import (
|
|||||||
"sigs.k8s.io/yaml"
|
"sigs.k8s.io/yaml"
|
||||||
)
|
)
|
||||||
|
|
||||||
// OpenAPIDefinition describes single type.
|
|
||||||
// Normally these definitions are auto-generated using gen-openapi.
|
|
||||||
// Same as in k8s.io / kube-openapi / pkg / common.
|
|
||||||
type OpenAPIDefinition struct {
|
|
||||||
Schema spec.Schema
|
|
||||||
Dependencies []string
|
|
||||||
}
|
|
||||||
|
|
||||||
type myProperties map[string]spec.Schema
|
type myProperties map[string]spec.Schema
|
||||||
type nameToApiMap map[string]OpenAPIDefinition
|
type nameToApiMap map[string]common.OpenAPIDefinition
|
||||||
|
|
||||||
// LoadConfigFromCRDs parse CRD schemas from paths into a TransformerConfig
|
// LoadConfigFromCRDs parse CRD schemas from paths into a TransformerConfig
|
||||||
func LoadConfigFromCRDs(
|
func LoadConfigFromCRDs(
|
||||||
@@ -169,7 +162,7 @@ func loadCrdIntoConfig(
|
|||||||
err = theConfig.AddNamereferenceFieldSpec(
|
err = theConfig.AddNamereferenceFieldSpec(
|
||||||
builtinconfig.NameBackReferences{
|
builtinconfig.NameBackReferences{
|
||||||
Gvk: resid.Gvk{Kind: kind, Version: version},
|
Gvk: resid.Gvk{Kind: kind, Version: version},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
makeFs(theGvk, append(path, propName, nameKey))},
|
makeFs(theGvk, append(path, propName, nameKey))},
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -8,9 +8,10 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
|
"sigs.k8s.io/kustomize/api/loader"
|
||||||
|
|
||||||
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
. "sigs.k8s.io/kustomize/api/internal/accumulator"
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/loader"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
)
|
)
|
||||||
@@ -139,19 +140,21 @@ func TestLoadCRDs(t *testing.T) {
|
|||||||
nbrs := []builtinconfig.NameBackReferences{
|
nbrs := []builtinconfig.NameBackReferences{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "Secret", Version: "v1"},
|
Gvk: resid.Gvk{Kind: "Secret", Version: "v1"},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "MyKind"},
|
CreateIfNotPresent: false,
|
||||||
Path: "spec/secretRef/name",
|
Gvk: resid.Gvk{Kind: "MyKind"},
|
||||||
|
Path: "spec/secretRef/name",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "Bee", Version: "v1beta1"},
|
Gvk: resid.Gvk{Kind: "Bee", Version: "v1beta1"},
|
||||||
Referrers: []types.FieldSpec{
|
FieldSpecs: []types.FieldSpec{
|
||||||
{
|
{
|
||||||
Gvk: resid.Gvk{Kind: "MyKind"},
|
CreateIfNotPresent: false,
|
||||||
Path: "spec/beeRef/name",
|
Gvk: resid.Gvk{Kind: "MyKind"},
|
||||||
|
Path: "spec/beeRef/name",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -7,26 +7,22 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filters/nameref"
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
"sigs.k8s.io/kustomize/api/resource"
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
|
"sigs.k8s.io/kustomize/api/transform"
|
||||||
)
|
)
|
||||||
|
|
||||||
type nameReferenceTransformer struct {
|
type nameReferenceTransformer struct {
|
||||||
backRefs []builtinconfig.NameBackReferences
|
backRefs []builtinconfig.NameBackReferences
|
||||||
}
|
}
|
||||||
|
|
||||||
const doDebug = false
|
|
||||||
|
|
||||||
var _ resmap.Transformer = &nameReferenceTransformer{}
|
var _ resmap.Transformer = &nameReferenceTransformer{}
|
||||||
|
|
||||||
type filterMap map[*resource.Resource][]nameref.Filter
|
|
||||||
|
|
||||||
// newNameReferenceTransformer constructs a nameReferenceTransformer
|
// newNameReferenceTransformer constructs a nameReferenceTransformer
|
||||||
// with a given slice of NameBackReferences.
|
// with a given slice of NameBackReferences.
|
||||||
func newNameReferenceTransformer(
|
func newNameReferenceTransformer(br []builtinconfig.NameBackReferences) resmap.Transformer {
|
||||||
br []builtinconfig.NameBackReferences) resmap.Transformer {
|
|
||||||
if br == nil {
|
if br == nil {
|
||||||
log.Fatal("backrefs not expected to be nil")
|
log.Fatal("backrefs not expected to be nil")
|
||||||
}
|
}
|
||||||
@@ -39,61 +35,13 @@ func newNameReferenceTransformer(
|
|||||||
//
|
//
|
||||||
// For example, a HorizontalPodAutoscaler (HPA)
|
// For example, a HorizontalPodAutoscaler (HPA)
|
||||||
// necessarily refers to a Deployment, the thing that
|
// necessarily refers to a Deployment, the thing that
|
||||||
// an HPA scales. In this case:
|
// the HPA scales. The Deployment name might change
|
||||||
|
// (e.g. prefix added), and the reference in the HPA
|
||||||
|
// has to be fixed.
|
||||||
//
|
//
|
||||||
// - the HPA instance is the Referrer,
|
// In the outer loop over the ResMap below, say we
|
||||||
// - the Deployment instance is the ReferralTarget.
|
// encounter a specific HPA. Then, in scanning backrefs,
|
||||||
//
|
// we encounter an entry like
|
||||||
// If the Deployment's name changes, e.g. a prefix is added,
|
|
||||||
// then the HPA's reference to the Deployment must be fixed.
|
|
||||||
//
|
|
||||||
func (t *nameReferenceTransformer) Transform(m resmap.ResMap) error {
|
|
||||||
fMap := t.determineFilters(m.Resources())
|
|
||||||
debug(fMap)
|
|
||||||
for r, fList := range fMap {
|
|
||||||
c := m.SubsetThatCouldBeReferencedByResource(r)
|
|
||||||
for _, f := range fList {
|
|
||||||
f.Referrer = r
|
|
||||||
f.ReferralCandidates = c
|
|
||||||
if err := f.Referrer.ApplyFilter(f); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func debug(fMap filterMap) {
|
|
||||||
if !doDebug {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
fmt.Printf("filterMap has %d entries:\n", len(fMap))
|
|
||||||
rCount := 0
|
|
||||||
for r, fList := range fMap {
|
|
||||||
yml, _ := r.AsYAML()
|
|
||||||
rCount++
|
|
||||||
fmt.Printf(`
|
|
||||||
---- %3d. possible referrer -------------
|
|
||||||
%s
|
|
||||||
---------`, rCount, string(yml),
|
|
||||||
)
|
|
||||||
for i, f := range fList {
|
|
||||||
fmt.Printf(`
|
|
||||||
%3d/%3d update: %s
|
|
||||||
from: %s
|
|
||||||
`, rCount, i+1, f.NameFieldToUpdate.Path, f.ReferralTarget,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Produce a map from referrer resources that might need to be fixed
|
|
||||||
// to filters that might fix them. The keys to this map are potential
|
|
||||||
// referrers, so won't include resources like ConfigMap or Secret.
|
|
||||||
//
|
|
||||||
// In the inner loop over the resources below, say we
|
|
||||||
// encounter an HPA instance. Then, in scanning the set
|
|
||||||
// of all known backrefs, we encounter an entry like
|
|
||||||
//
|
//
|
||||||
// - kind: Deployment
|
// - kind: Deployment
|
||||||
// fieldSpecs:
|
// fieldSpecs:
|
||||||
@@ -102,53 +50,220 @@ func debug(fMap filterMap) {
|
|||||||
//
|
//
|
||||||
// This entry says that an HPA, via its
|
// This entry says that an HPA, via its
|
||||||
// 'spec/scaleTargetRef/name' field, may refer to a
|
// 'spec/scaleTargetRef/name' field, may refer to a
|
||||||
// Deployment.
|
// Deployment. This match to HPA means we may need to
|
||||||
|
// modify the value in its 'spec/scaleTargetRef/name'
|
||||||
|
// field, by searching for the thing it refers to,
|
||||||
|
// and getting its new name.
|
||||||
//
|
//
|
||||||
// This means that a filter will need to hunt for the right Deployment,
|
// As a filter, and search optimization, we compute a
|
||||||
// obtain it's new name, and write that name into the HPA's
|
// subset of all resources that the HPA could refer to,
|
||||||
// 'spec/scaleTargetRef/name' field. Return a filter that can do that.
|
// by excluding objects from other namespaces, and
|
||||||
func (t *nameReferenceTransformer) determineFilters(
|
// excluding objects that don't have the same prefix-
|
||||||
resources []*resource.Resource) (fMap filterMap) {
|
// suffix mods as the HPA.
|
||||||
fMap = make(filterMap)
|
//
|
||||||
for _, backReference := range t.backRefs {
|
// We look in this subset for all Deployment objects
|
||||||
for _, referrerSpec := range backReference.Referrers {
|
// with a resId that has a Name matching the field value
|
||||||
for _, res := range resources {
|
// present in the HPA. If no match do nothing; if more
|
||||||
if res.OrgId().IsSelected(&referrerSpec.Gvk) {
|
// than one match, it's an error.
|
||||||
// If this is true, the res might be a referrer, and if
|
//
|
||||||
// so, the name reference it holds might need an update.
|
// We overwrite the HPA name field with the value found
|
||||||
if resHasField(res, referrerSpec.Path) {
|
// in the Deployment's name field (the name in the raw
|
||||||
// Optimization - the referrer has the field
|
// object - the modified name - not the unmodified name
|
||||||
// that might need updating.
|
// in the Deployment's resId).
|
||||||
fMap[res] = append(fMap[res], nameref.Filter{
|
//
|
||||||
// Name field to write in the Referrer.
|
// This process assumes that the name stored in a ResId
|
||||||
// If the path specified here isn't found in
|
// (the ResMap key) isn't modified by name transformers.
|
||||||
// the Referrer, nothing happens (no error,
|
// Name transformers should only modify the name in the
|
||||||
// no field creation).
|
// body of the resource object (the value in the ResMap).
|
||||||
NameFieldToUpdate: referrerSpec,
|
//
|
||||||
// Specification of object class to read from.
|
func (o *nameReferenceTransformer) Transform(m resmap.ResMap) error {
|
||||||
// Always read from metadata/name field.
|
// TODO: Too much looping, here and in transitive calls.
|
||||||
ReferralTarget: backReference.Gvk,
|
for _, referrer := range m.Resources() {
|
||||||
})
|
var candidates resmap.ResMap
|
||||||
|
for _, target := range o.backRefs {
|
||||||
|
for _, fSpec := range target.FieldSpecs {
|
||||||
|
if referrer.OrgId().IsSelected(&fSpec.Gvk) {
|
||||||
|
if candidates == nil {
|
||||||
|
candidates = m.SubsetThatCouldBeReferencedByResource(referrer)
|
||||||
|
}
|
||||||
|
err := transform.MutateField(
|
||||||
|
referrer.Map(),
|
||||||
|
fSpec.PathSlice(),
|
||||||
|
fSpec.CreateIfNotPresent,
|
||||||
|
o.getNewNameFunc(
|
||||||
|
// referrer could be an HPA instance,
|
||||||
|
// target could be Gvk for Deployment,
|
||||||
|
// candidate a list of resources "reachable"
|
||||||
|
// from the HPA.
|
||||||
|
referrer, target.Gvk, candidates))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return fMap
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: check res for field existence here to avoid extra work.
|
// selectReferral picks the referral among a subset of candidates.
|
||||||
// res.GetFieldValue, which uses yaml.Lookup under the hood, doesn't know
|
// It returns the current name and namespace of the selected candidate.
|
||||||
// how to parse fieldspec-style paths that make no distinction
|
// Note that the content of the referricalCandidateSubset slice is most of the time
|
||||||
// between maps and sequences. This means it cannot lookup commonly
|
// identical to the referralCandidates resmap. Still in some cases, such
|
||||||
// used "indeterminate" paths like
|
// as ClusterRoleBinding, the subset only contains the resources of a specific
|
||||||
// spec/containers/env/valueFrom/configMapKeyRef/name
|
// namespace.
|
||||||
// ('containers' is a list, not a map).
|
func (o *nameReferenceTransformer) selectReferral(
|
||||||
// However, the fieldspec filter does know how to handle this;
|
oldName string,
|
||||||
// extract that code and call it here?
|
referrer *resource.Resource,
|
||||||
func resHasField(res *resource.Resource, path string) bool {
|
target resid.Gvk,
|
||||||
return true
|
referralCandidates resmap.ResMap,
|
||||||
// fld := strings.Join(utils.PathSplitter(path), ".")
|
referralCandidateSubset []*resource.Resource) (interface{}, interface{}, error) {
|
||||||
// _, e := res.GetFieldValue(fld)
|
|
||||||
// return e == nil
|
for _, res := range referralCandidateSubset {
|
||||||
|
id := res.OrgId()
|
||||||
|
if id.IsSelected(&target) && res.GetOriginalName() == oldName {
|
||||||
|
matches := referralCandidates.GetMatchingResourcesByOriginalId(id.Equals)
|
||||||
|
// If there's more than one match, there's no way
|
||||||
|
// to know which one to pick, so emit error.
|
||||||
|
if len(matches) > 1 {
|
||||||
|
return nil, nil, fmt.Errorf(
|
||||||
|
"multiple matches for %s:\n %v",
|
||||||
|
id, getIds(matches))
|
||||||
|
}
|
||||||
|
// In the resource, note that it is referenced
|
||||||
|
// by the referrer.
|
||||||
|
res.AppendRefBy(referrer.CurId())
|
||||||
|
// Return transformed name of the object,
|
||||||
|
// complete with prefixes, hashes, etc.
|
||||||
|
return res.GetName(), res.GetNamespace(), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return oldName, nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// utility function to replace a simple string by the new name
|
||||||
|
func (o *nameReferenceTransformer) getSimpleNameField(
|
||||||
|
oldName string,
|
||||||
|
referrer *resource.Resource,
|
||||||
|
target resid.Gvk,
|
||||||
|
referralCandidates resmap.ResMap,
|
||||||
|
referralCandidateSubset []*resource.Resource) (interface{}, error) {
|
||||||
|
|
||||||
|
newName, _, err := o.selectReferral(oldName, referrer, target,
|
||||||
|
referralCandidates, referralCandidateSubset)
|
||||||
|
|
||||||
|
return newName, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// utility function to replace name field within a map[string]interface{}
|
||||||
|
// and leverage the namespace field.
|
||||||
|
func (o *nameReferenceTransformer) getNameAndNsStruct(
|
||||||
|
inMap map[string]interface{},
|
||||||
|
referrer *resource.Resource,
|
||||||
|
target resid.Gvk,
|
||||||
|
referralCandidates resmap.ResMap) (interface{}, error) {
|
||||||
|
|
||||||
|
// Example:
|
||||||
|
if _, ok := inMap["name"]; !ok {
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"%#v is expected to contain a name field", inMap)
|
||||||
|
}
|
||||||
|
oldName, ok := inMap["name"].(string)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"%#v is expected to contain a name field of type string", oldName)
|
||||||
|
}
|
||||||
|
|
||||||
|
subset := referralCandidates.Resources()
|
||||||
|
if namespacevalue, ok := inMap["namespace"]; ok {
|
||||||
|
namespace := namespacevalue.(string)
|
||||||
|
bynamespace := referralCandidates.GroupedByOriginalNamespace()
|
||||||
|
if _, ok := bynamespace[namespace]; !ok {
|
||||||
|
return inMap, nil
|
||||||
|
}
|
||||||
|
subset = bynamespace[namespace]
|
||||||
|
}
|
||||||
|
|
||||||
|
newname, newnamespace, err := o.selectReferral(oldName, referrer, target,
|
||||||
|
referralCandidates, subset)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newname == oldName) && (newnamespace == nil) {
|
||||||
|
// no candidate found.
|
||||||
|
return inMap, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
inMap["name"] = newname
|
||||||
|
if newnamespace != "" {
|
||||||
|
// We don't want value "" to replace value "default" since
|
||||||
|
// the empty string is handled as a wild card here not default namespace
|
||||||
|
// by kubernetes.
|
||||||
|
inMap["namespace"] = newnamespace
|
||||||
|
}
|
||||||
|
return inMap, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *nameReferenceTransformer) getNewNameFunc(
|
||||||
|
referrer *resource.Resource,
|
||||||
|
target resid.Gvk,
|
||||||
|
referralCandidates resmap.ResMap) func(in interface{}) (interface{}, error) {
|
||||||
|
return func(in interface{}) (interface{}, error) {
|
||||||
|
switch thing := in.(type) {
|
||||||
|
case string:
|
||||||
|
return o.getSimpleNameField(thing, referrer, target,
|
||||||
|
referralCandidates, referralCandidates.Resources())
|
||||||
|
case map[string]interface{}:
|
||||||
|
// Kind: ValidatingWebhookConfiguration
|
||||||
|
// FieldSpec is webhooks/clientConfig/service
|
||||||
|
return o.getNameAndNsStruct(thing, referrer, target,
|
||||||
|
referralCandidates)
|
||||||
|
case []interface{}:
|
||||||
|
for idx, item := range thing {
|
||||||
|
switch value := item.(type) {
|
||||||
|
case string:
|
||||||
|
// Kind: Role/ClusterRole
|
||||||
|
// FieldSpec is rules.resourceNames
|
||||||
|
newName, err := o.getSimpleNameField(value, referrer, target,
|
||||||
|
referralCandidates, referralCandidates.Resources())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
thing[idx] = newName
|
||||||
|
case map[string]interface{}:
|
||||||
|
// Kind: RoleBinding/ClusterRoleBinding
|
||||||
|
// FieldSpec is subjects
|
||||||
|
// Note: The corresponding fieldSpec had been changed from
|
||||||
|
// from path: subjects/name to just path: subjects. This is
|
||||||
|
// what get mutatefield to request the mapping of the whole
|
||||||
|
// map containing namespace and name instead of just a simple
|
||||||
|
// string field containing the name
|
||||||
|
newMap, err := o.getNameAndNsStruct(value, referrer, target,
|
||||||
|
referralCandidates)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
thing[idx] = newMap
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"%#v is expected to be either a []string or a []map[string]interface{}", in)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return in, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf(
|
||||||
|
"%#v is expected to be either a string or a []interface{}", in)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getIds(rs []*resource.Resource) []string {
|
||||||
|
var result []string
|
||||||
|
for _, r := range rs {
|
||||||
|
result = append(result, r.CurId().String()+"\n")
|
||||||
|
}
|
||||||
|
return result
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,17 +8,17 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
"sigs.k8s.io/kustomize/api/internal/plugins/builtinconfig"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/k8sdeps/kunstruct"
|
||||||
"sigs.k8s.io/kustomize/api/provider"
|
|
||||||
"sigs.k8s.io/kustomize/api/resid"
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
"sigs.k8s.io/kustomize/api/resmap"
|
"sigs.k8s.io/kustomize/api/resmap"
|
||||||
|
"sigs.k8s.io/kustomize/api/resource"
|
||||||
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
resmaptest_test "sigs.k8s.io/kustomize/api/testutils/resmaptest"
|
||||||
)
|
)
|
||||||
|
|
||||||
const notEqualErrFmt = "expected (self) doesn't match actual (other): %v"
|
|
||||||
|
|
||||||
func TestNameReferenceHappyRun(t *testing.T) {
|
func TestNameReferenceHappyRun(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).AddWithName(
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).AddWithName(
|
||||||
"cm1",
|
"cm1",
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -220,7 +220,6 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
"secret1",
|
"secret1",
|
||||||
"secret1",
|
"secret1",
|
||||||
"secret2",
|
"secret2",
|
||||||
"cm1",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -262,8 +261,7 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}).ResMap()
|
}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).ReplaceResource(
|
||||||
t, m.ShallowCopy()).ReplaceResource(
|
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"group": "apps",
|
"group": "apps",
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -424,7 +422,6 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
"someprefix-secret1-somehash",
|
"someprefix-secret1-somehash",
|
||||||
"someprefix-secret1-somehash",
|
"someprefix-secret1-somehash",
|
||||||
"secret2",
|
"secret2",
|
||||||
"someprefix-cm1-somehash",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -473,17 +470,19 @@ func TestNameReferenceHappyRun(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNameReferenceUnhappyRun(t *testing.T) {
|
func TestNameReferenceUnhappyRun(t *testing.T) {
|
||||||
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
tests := []struct {
|
tests := []struct {
|
||||||
resMap resmap.ResMap
|
resMap resmap.ResMap
|
||||||
expectedErr string
|
expectedErr string
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
resMap: resmaptest_test.NewRmBuilderDefault(t).Add(
|
resMap: resmaptest_test.NewRmBuilder(t, rf).Add(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
@@ -503,7 +502,7 @@ func TestNameReferenceUnhappyRun(t *testing.T) {
|
|||||||
}).ResMap(),
|
}).ResMap(),
|
||||||
expectedErr: "is expected to be"},
|
expectedErr: "is expected to be"},
|
||||||
{
|
{
|
||||||
resMap: resmaptest_test.NewRmBuilderDefault(t).Add(
|
resMap: resmaptest_test.NewRmBuilder(t, rf).Add(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
@@ -521,27 +520,7 @@ func TestNameReferenceUnhappyRun(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
}).ResMap(),
|
}).ResMap(),
|
||||||
// TODO(#3304): DECISION - kyaml better; not a bug.
|
expectedErr: "is expected to contain a name field"},
|
||||||
expectedErr: konfig.IfApiMachineryElseKyaml(
|
|
||||||
`updating name reference in 'rules/resourceNames' field of `+
|
|
||||||
`'rbac.authorization.k8s.io_v1_ClusterRole|~X|cr'`+
|
|
||||||
`: considering field 'rules/resourceNames' of object
|
|
||||||
{"apiVersion": "rbac.authorization.k8s.io/v1", "kind": "ClusterRole", "metadata": {
|
|
||||||
"name": "cr"}, "rules": [{"resourceNames": {"foo": "bar"}, "resources": ["secrets"]}]}
|
|
||||||
: visit traversal on path: [resourceNames]: path config error; no 'name' field in node`,
|
|
||||||
`updating name reference in 'rules/resourceNames' field of `+
|
|
||||||
`'rbac.authorization.k8s.io_v1_ClusterRole|~X|cr'`+
|
|
||||||
`: considering field 'rules/resourceNames' of object
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
kind: ClusterRole
|
|
||||||
metadata:
|
|
||||||
name: cr
|
|
||||||
rules:
|
|
||||||
- resourceNames:
|
|
||||||
foo: bar
|
|
||||||
resources:
|
|
||||||
- secrets
|
|
||||||
: visit traversal on path: [resourceNames]: path config error; no 'name' field in node`)},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nrt := newNameReferenceTransformer(builtinconfig.MakeDefaultConfig().NameReference)
|
nrt := newNameReferenceTransformer(builtinconfig.MakeDefaultConfig().NameReference)
|
||||||
@@ -552,14 +531,15 @@ rules:
|
|||||||
}
|
}
|
||||||
|
|
||||||
if !strings.Contains(err.Error(), test.expectedErr) {
|
if !strings.Contains(err.Error(), test.expectedErr) {
|
||||||
t.Fatalf("Incorrect error.\nExpected:\n %s\nGot:\n%v",
|
t.Fatalf("Incorrect error.\nExpected: %s, but got %v",
|
||||||
test.expectedErr, err)
|
test.expectedErr, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
||||||
rf := provider.NewDefaultDepProvider().GetResourceFactory()
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
|
||||||
v1 := rf.FromMapWithName(
|
v1 := rf.FromMapWithName(
|
||||||
"volume1",
|
"volume1",
|
||||||
@@ -610,7 +590,7 @@ func TestNameReferencePersistentVolumeHappyRun(t *testing.T) {
|
|||||||
v2.AppendRefBy(c2.CurId())
|
v2.AppendRefBy(c2.CurId())
|
||||||
|
|
||||||
if err := m1.ErrorIfNotEqualLists(m2); err != nil {
|
if err := m1.ErrorIfNotEqualLists(m2); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -684,7 +664,9 @@ const (
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceNamespace(t *testing.T) {
|
func TestNameReferenceNamespace(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
// Add ConfigMap with the same org name in noNs, "ns1" and "ns2" namespaces
|
// Add ConfigMap with the same org name in noNs, "ns1" and "ns2" namespaces
|
||||||
AddWithName(orgname, map[string]interface{}{
|
AddWithName(orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -733,7 +715,7 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
AddWithNsAndName(ns1, orgname, deploymentMap(ns1, prefixedname, orgname, orgname)).
|
AddWithNsAndName(ns1, orgname, deploymentMap(ns1, prefixedname, orgname, orgname)).
|
||||||
AddWithNsAndName(ns2, orgname, deploymentMap(ns2, suffixedname, orgname, orgname)).ResMap()
|
AddWithNsAndName(ns2, orgname, deploymentMap(ns2, suffixedname, orgname, orgname)).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(deploymentMap(defaultNs, modifiedname, modifiedname, modifiedname)).
|
ReplaceResource(deploymentMap(defaultNs, modifiedname, modifiedname, modifiedname)).
|
||||||
ReplaceResource(deploymentMap(ns1, prefixedname, prefixedname, prefixedname)).
|
ReplaceResource(deploymentMap(ns1, prefixedname, prefixedname, prefixedname)).
|
||||||
ReplaceResource(deploymentMap(ns2, suffixedname, suffixedname, suffixedname)).ResMap()
|
ReplaceResource(deploymentMap(ns2, suffixedname, suffixedname, suffixedname)).ResMap()
|
||||||
@@ -744,9 +726,8 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -754,7 +735,9 @@ func TestNameReferenceNamespace(t *testing.T) {
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceClusterWide(t *testing.T) {
|
func TestNameReferenceClusterWide(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
// Add ServiceAccount with the same org name in noNs, "ns1" and "ns2" namespaces
|
// Add ServiceAccount with the same org name in noNs, "ns1" and "ns2" namespaces
|
||||||
AddWithName(orgname, map[string]interface{}{
|
AddWithName(orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
@@ -806,9 +789,9 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
"roleRef": map[string]interface{}{
|
"roleRef": map[string]interface{}{
|
||||||
"apiGroup": "rbac.authorization.k8s.io",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
"name": orgname,
|
"name": orgname,
|
||||||
},
|
},
|
||||||
"subjects": []interface{}{
|
"subjects": []interface{}{
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
@@ -833,7 +816,7 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}}).ResMap()
|
}}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(
|
ReplaceResource(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
@@ -862,9 +845,9 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
"roleRef": map[string]interface{}{
|
"roleRef": map[string]interface{}{
|
||||||
"apiGroup": "rbac.authorization.k8s.io",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
// The following tests required a change in
|
// The following tests required a change in
|
||||||
// getNameFunc implementation in order to leverage
|
// getNameFunc implementation in order to leverage
|
||||||
@@ -906,9 +889,8 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -916,7 +898,9 @@ func TestNameReferenceClusterWide(t *testing.T) {
|
|||||||
// object with the same original names (uniquename) in different namespaces
|
// object with the same original names (uniquename) in different namespaces
|
||||||
// and with different current Id.
|
// and with different current Id.
|
||||||
func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
AddWithNsAndName(ns4, orgname, map[string]interface{}{
|
AddWithNsAndName(ns4, orgname, map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "Secret",
|
"kind": "Secret",
|
||||||
@@ -953,9 +937,9 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
"roleRef": map[string]interface{}{
|
"roleRef": map[string]interface{}{
|
||||||
"apiGroup": "rbac.authorization.k8s.io",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
"name": orgname,
|
"name": orgname,
|
||||||
},
|
},
|
||||||
"subjects": []interface{}{
|
"subjects": []interface{}{
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
@@ -980,7 +964,7 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
},
|
},
|
||||||
}}).ResMap()
|
}}).ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(
|
ReplaceResource(
|
||||||
map[string]interface{}{
|
map[string]interface{}{
|
||||||
"apiVersion": "rbac.authorization.k8s.io/v1",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
@@ -989,9 +973,9 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
"roleRef": map[string]interface{}{
|
"roleRef": map[string]interface{}{
|
||||||
"apiGroup": "rbac.authorization.k8s.io",
|
"apiVersion": "rbac.authorization.k8s.io/v1",
|
||||||
"kind": "ClusterRole",
|
"kind": "ClusterRole",
|
||||||
"name": modifiedname,
|
"name": modifiedname,
|
||||||
},
|
},
|
||||||
// The following tests required a change in
|
// The following tests required a change in
|
||||||
// getNameFunc implementation in order to leverage
|
// getNameFunc implementation in order to leverage
|
||||||
@@ -1033,9 +1017,8 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1043,7 +1026,9 @@ func TestNameReferenceNamespaceTransformation(t *testing.T) {
|
|||||||
// It validates the change done is IsSameFuzzyNamespace which
|
// It validates the change done is IsSameFuzzyNamespace which
|
||||||
// uses the IsNsEquals method instead of the simple == operator.
|
// uses the IsNsEquals method instead of the simple == operator.
|
||||||
func TestNameReferenceCandidateSelection(t *testing.T) {
|
func TestNameReferenceCandidateSelection(t *testing.T) {
|
||||||
m := resmaptest_test.NewRmBuilderDefault(t).
|
rf := resource.NewFactory(
|
||||||
|
kunstruct.NewKunstructuredFactoryImpl())
|
||||||
|
m := resmaptest_test.NewRmBuilder(t, rf).
|
||||||
AddWithName("cm1", map[string]interface{}{
|
AddWithName("cm1", map[string]interface{}{
|
||||||
"apiVersion": "v1",
|
"apiVersion": "v1",
|
||||||
"kind": "ConfigMap",
|
"kind": "ConfigMap",
|
||||||
@@ -1060,7 +1045,7 @@ func TestNameReferenceCandidateSelection(t *testing.T) {
|
|||||||
AddWithName("deploy1", deploymentMap("", "p1-deploy1", "cm1", "secret1")).
|
AddWithName("deploy1", deploymentMap("", "p1-deploy1", "cm1", "secret1")).
|
||||||
ResMap()
|
ResMap()
|
||||||
|
|
||||||
expected := resmaptest_test.NewSeededRmBuilderDefault(t, m.ShallowCopy()).
|
expected := resmaptest_test.NewSeededRmBuilder(t, rf, m.ShallowCopy()).
|
||||||
ReplaceResource(deploymentMap("", "p1-deploy1", "p1-cm1-hash", "p1-secret1-hash")).
|
ReplaceResource(deploymentMap("", "p1-deploy1", "p1-cm1-hash", "p1-secret1-hash")).
|
||||||
ResMap()
|
ResMap()
|
||||||
|
|
||||||
@@ -1070,8 +1055,7 @@ func TestNameReferenceCandidateSelection(t *testing.T) {
|
|||||||
t.Fatalf("unexpected error: %v", err)
|
t.Fatalf("unexpected error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.RemoveBuildAnnotations()
|
|
||||||
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
if err = expected.ErrorIfNotEqualLists(m); err != nil {
|
||||||
t.Fatalf(notEqualErrFmt, err)
|
t.Fatalf("actual doesn't match expected: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user