mirror of
https://github.com/kubernetes-sigs/kustomize.git
synced 2026-05-17 18:25:26 +00:00
Merge pull request #3129 from Shell32-Natsu/patches-command
Update edit patch command
This commit is contained in:
@@ -17,3 +17,12 @@ type Patch struct {
|
|||||||
// Target points to the resources that the patch is applied to
|
// Target points to the resources that the patch is applied to
|
||||||
Target *Selector `json:"target,omitempty" yaml:"target,omitempty"`
|
Target *Selector `json:"target,omitempty" yaml:"target,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Equals return true if p equals o.
|
||||||
|
func (p *Patch) Equals(o Patch) bool {
|
||||||
|
targetEqual := (p.Target == o.Target) ||
|
||||||
|
(p.Target != nil && o.Target != nil && *p.Target == *o.Target)
|
||||||
|
return p.Path == o.Path &&
|
||||||
|
p.Patch == o.Patch &&
|
||||||
|
targetEqual
|
||||||
|
}
|
||||||
|
|||||||
125
api/types/patch_test.go
Normal file
125
api/types/patch_test.go
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
package types_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"sigs.k8s.io/kustomize/api/resid"
|
||||||
|
. "sigs.k8s.io/kustomize/api/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestPatchEquals(t *testing.T) {
|
||||||
|
selector := Selector{
|
||||||
|
Gvk: resid.Gvk{
|
||||||
|
Group: "group",
|
||||||
|
Version: "version",
|
||||||
|
Kind: "kind",
|
||||||
|
},
|
||||||
|
Name: "name",
|
||||||
|
Namespace: "namespace",
|
||||||
|
LabelSelector: "selector",
|
||||||
|
AnnotationSelector: "selector",
|
||||||
|
}
|
||||||
|
type testcase struct {
|
||||||
|
patch1 Patch
|
||||||
|
patch2 Patch
|
||||||
|
expect bool
|
||||||
|
name string
|
||||||
|
}
|
||||||
|
testcases := []testcase{
|
||||||
|
{
|
||||||
|
name: "empty patches",
|
||||||
|
patch1: Patch{},
|
||||||
|
patch2: Patch{},
|
||||||
|
expect: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "full patches",
|
||||||
|
patch1: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
Target: &Selector{
|
||||||
|
Gvk: resid.Gvk{
|
||||||
|
Group: "group",
|
||||||
|
Version: "version",
|
||||||
|
Kind: "kind",
|
||||||
|
},
|
||||||
|
Name: "name",
|
||||||
|
Namespace: "namespace",
|
||||||
|
LabelSelector: "selector",
|
||||||
|
AnnotationSelector: "selector",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
patch2: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
Target: &Selector{
|
||||||
|
Gvk: resid.Gvk{
|
||||||
|
Group: "group",
|
||||||
|
Version: "version",
|
||||||
|
Kind: "kind",
|
||||||
|
},
|
||||||
|
Name: "name",
|
||||||
|
Namespace: "namespace",
|
||||||
|
LabelSelector: "selector",
|
||||||
|
AnnotationSelector: "selector",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
expect: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "same target",
|
||||||
|
patch1: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
Target: &selector,
|
||||||
|
},
|
||||||
|
patch2: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
Target: &selector,
|
||||||
|
},
|
||||||
|
expect: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "omit target",
|
||||||
|
patch1: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
},
|
||||||
|
patch2: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
},
|
||||||
|
expect: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "one nil target",
|
||||||
|
patch1: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
Target: &selector,
|
||||||
|
},
|
||||||
|
patch2: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
Patch: "bar",
|
||||||
|
},
|
||||||
|
expect: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "different path",
|
||||||
|
patch1: Patch{
|
||||||
|
Path: "foo",
|
||||||
|
},
|
||||||
|
patch2: Patch{
|
||||||
|
Path: "bar",
|
||||||
|
},
|
||||||
|
expect: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testcases {
|
||||||
|
if tc.expect != tc.patch1.Equals(tc.patch2) {
|
||||||
|
t.Fatalf("%s: unexpected result %v", tc.name, !tc.expect)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,13 +4,13 @@
|
|||||||
|
|
||||||
## ConfigMap generation and rolling updates
|
## ConfigMap generation and rolling updates
|
||||||
|
|
||||||
Kustomize provides two ways of adding ConfigMap in one `kustomization`, either by declaring ConfigMap as a [resource] or declaring ConfigMap from a ConfigMapGenerator. The formats inside `kustomization.yaml` are
|
Kustomize provides two ways of adding ConfigMap in one `kustomization`, either by declaring ConfigMap as a [resource] or declaring ConfigMap from a ConfigMapGenerator. The formats inside `kustomization.yaml` are
|
||||||
|
|
||||||
> ```
|
> ```
|
||||||
> # declare ConfigMap as a resource
|
> # declare ConfigMap as a resource
|
||||||
> resources:
|
> resources:
|
||||||
> - configmap.yaml
|
> - configmap.yaml
|
||||||
>
|
>
|
||||||
> # declare ConfigMap from a ConfigMapGenerator
|
> # declare ConfigMap from a ConfigMapGenerator
|
||||||
> configMapGenerator:
|
> configMapGenerator:
|
||||||
> - name: a-configmap
|
> - name: a-configmap
|
||||||
@@ -28,7 +28,9 @@ In this demo, the same [hello_world](helloWorld/README.md) is used while the Con
|
|||||||
### Establish base and staging
|
### Establish base and staging
|
||||||
|
|
||||||
Establish the base with a configMapGenerator
|
Establish the base with a configMapGenerator
|
||||||
|
|
||||||
<!-- @establishBase @testAgainstLatestRelease -->
|
<!-- @establishBase @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
DEMO_HOME=$(mktemp -d)
|
DEMO_HOME=$(mktemp -d)
|
||||||
|
|
||||||
@@ -46,16 +48,18 @@ commonLabels:
|
|||||||
resources:
|
resources:
|
||||||
- deployment.yaml
|
- deployment.yaml
|
||||||
- service.yaml
|
- service.yaml
|
||||||
configMapGenerator:
|
configMapGenerator:
|
||||||
- name: the-map
|
- name: the-map
|
||||||
literals:
|
literals:
|
||||||
- altGreeting=Good Morning!
|
- altGreeting=Good Morning!
|
||||||
- enableRisky="false"
|
- enableRisky="false"
|
||||||
EOF
|
EOF
|
||||||
```
|
```
|
||||||
|
|
||||||
Establish the staging with a patch applied to the ConfigMap
|
Establish the staging with a patch applied to the ConfigMap
|
||||||
|
|
||||||
<!-- @establishStaging @testAgainstLatestRelease -->
|
<!-- @establishStaging @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
OVERLAYS=$DEMO_HOME/overlays
|
OVERLAYS=$DEMO_HOME/overlays
|
||||||
mkdir -p $OVERLAYS/staging
|
mkdir -p $OVERLAYS/staging
|
||||||
@@ -92,8 +96,8 @@ configured with data from a configMap.
|
|||||||
|
|
||||||
The deployment refers to this map by name:
|
The deployment refers to this map by name:
|
||||||
|
|
||||||
|
|
||||||
<!-- @showDeployment @testAgainstLatestRelease -->
|
<!-- @showDeployment @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
grep -C 2 configMapKeyRef $BASE/deployment.yaml
|
grep -C 2 configMapKeyRef $BASE/deployment.yaml
|
||||||
```
|
```
|
||||||
@@ -106,12 +110,12 @@ changed, so such updates have no effect.
|
|||||||
The recommended way to change a deployment's
|
The recommended way to change a deployment's
|
||||||
configuration is to
|
configuration is to
|
||||||
|
|
||||||
1. create a new configMap with a new name,
|
1. create a new configMap with a new name,
|
||||||
1. patch the _deployment_, modifying the name value of
|
1. patch the _deployment_, modifying the name value of
|
||||||
the appropriate `configMapKeyRef` field.
|
the appropriate `configMapKeyRef` field.
|
||||||
|
|
||||||
This latter change initiates rolling update to the pods
|
This latter change initiates rolling update to the pods
|
||||||
in the deployment. The older configMap, when no longer
|
in the deployment. The older configMap, when no longer
|
||||||
referenced by any other resource, is eventually [garbage
|
referenced by any other resource, is eventually [garbage
|
||||||
collected](/../../issues/242).
|
collected](/../../issues/242).
|
||||||
|
|
||||||
@@ -120,6 +124,7 @@ collected](/../../issues/242).
|
|||||||
The _staging_ [variant] here has a configMap [patch]:
|
The _staging_ [variant] here has a configMap [patch]:
|
||||||
|
|
||||||
<!-- @showMapPatch @testAgainstLatestRelease -->
|
<!-- @showMapPatch @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
cat $OVERLAYS/staging/map.yaml
|
cat $OVERLAYS/staging/map.yaml
|
||||||
```
|
```
|
||||||
@@ -131,6 +136,7 @@ resource spec.
|
|||||||
The ConfigMap it modifies is declared from a configMapGenerator.
|
The ConfigMap it modifies is declared from a configMapGenerator.
|
||||||
|
|
||||||
<!-- @showMapBase @testAgainstLatestRelease -->
|
<!-- @showMapBase @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
grep -C 4 configMapGenerator $BASE/kustomization.yaml
|
grep -C 4 configMapGenerator $BASE/kustomization.yaml
|
||||||
```
|
```
|
||||||
@@ -139,11 +145,12 @@ For a patch to work, the names in the `metadata/name`
|
|||||||
fields must match.
|
fields must match.
|
||||||
|
|
||||||
However, the name values specified in the file are
|
However, the name values specified in the file are
|
||||||
_not_ what gets used in the cluster. By design,
|
_not_ what gets used in the cluster. By design,
|
||||||
kustomize modifies names of ConfigMaps declared from ConfigMapGenerator. To see the names
|
kustomize modifies names of ConfigMaps declared from ConfigMapGenerator. To see the names
|
||||||
ultimately used in the cluster, just run kustomize:
|
ultimately used in the cluster, just run kustomize:
|
||||||
|
|
||||||
<!-- @grepStagingName @testAgainstLatestRelease -->
|
<!-- @grepStagingName @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging |\
|
kustomize build $OVERLAYS/staging |\
|
||||||
grep -B 8 -A 1 staging-the-map
|
grep -B 8 -A 1 staging-the-map
|
||||||
@@ -161,7 +168,8 @@ The suffix to the configMap name is generated from a
|
|||||||
hash of the maps content - in this case the name suffix
|
hash of the maps content - in this case the name suffix
|
||||||
is _5276h4th55_:
|
is _5276h4th55_:
|
||||||
|
|
||||||
<!-- @grepStagingHash -->
|
<!-- @grepStagingHash @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging | grep 5276h4th55
|
kustomize build $OVERLAYS/staging | grep 5276h4th55
|
||||||
```
|
```
|
||||||
@@ -170,6 +178,7 @@ Now modify the map patch, to change the greeting
|
|||||||
the server will use:
|
the server will use:
|
||||||
|
|
||||||
<!-- @changeMap @testAgainstLatestRelease -->
|
<!-- @changeMap @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
sed -i.bak 's/pineapple/kiwi/' $OVERLAYS/staging/map.yaml
|
sed -i.bak 's/pineapple/kiwi/' $OVERLAYS/staging/map.yaml
|
||||||
```
|
```
|
||||||
@@ -184,6 +193,7 @@ kustomize build $OVERLAYS/staging |\
|
|||||||
Run kustomize again to see the new configMap names:
|
Run kustomize again to see the new configMap names:
|
||||||
|
|
||||||
<!-- @grepStagingName @testAgainstLatestRelease -->
|
<!-- @grepStagingName @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging |\
|
kustomize build $OVERLAYS/staging |\
|
||||||
grep -B 8 -A 1 staging-the-map
|
grep -B 8 -A 1 staging-the-map
|
||||||
@@ -194,7 +204,8 @@ in three new names ending in _c2g8fcbf88_ - one in the
|
|||||||
configMap name itself, and two in the deployment that
|
configMap name itself, and two in the deployment that
|
||||||
uses the map:
|
uses the map:
|
||||||
|
|
||||||
<!-- @countHashes -->
|
<!-- @countHashes @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
test 3 == \
|
test 3 == \
|
||||||
$(kustomize build $OVERLAYS/staging | grep c2g8fcbf88 | wc -l); \
|
$(kustomize build $OVERLAYS/staging | grep c2g8fcbf88 | wc -l); \
|
||||||
@@ -204,7 +215,7 @@ test 3 == \
|
|||||||
Applying these resources to the cluster will result in
|
Applying these resources to the cluster will result in
|
||||||
a rolling update of the deployments pods, retargetting
|
a rolling update of the deployments pods, retargetting
|
||||||
them from the _5276h4th55_ maps to the _c2g8fcbf88_
|
them from the _5276h4th55_ maps to the _c2g8fcbf88_
|
||||||
maps. The system will later garbage collect the
|
maps. The system will later garbage collect the
|
||||||
unused maps.
|
unused maps.
|
||||||
|
|
||||||
## Rollback
|
## Rollback
|
||||||
|
|||||||
@@ -102,7 +102,7 @@ For Spring Boot application, we can set an active profile through the environmen
|
|||||||
the application will pick up an extra `application-<profile>.properties` file. With this, we can customize the configMap in two
|
the application will pick up an extra `application-<profile>.properties` file. With this, we can customize the configMap in two
|
||||||
steps. Add an environment variable through the patch and add a file to the configMap.
|
steps. Add an environment variable through the patch and add a file to the configMap.
|
||||||
|
|
||||||
<!-- @customizeConfigMap @testAgainstLatestRelease -->
|
<!-- @customizeConfigMap -->
|
||||||
```
|
```
|
||||||
cat <<EOF >$DEMO_HOME/patch.yaml
|
cat <<EOF >$DEMO_HOME/patch.yaml
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
@@ -119,7 +119,7 @@ spec:
|
|||||||
value: prod
|
value: prod
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
kustomize edit add patch patch.yaml
|
kustomize edit add patch --path patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
|
|
||||||
cat <<EOF >$DEMO_HOME/application-prod.properties
|
cat <<EOF >$DEMO_HOME/application-prod.properties
|
||||||
spring.jpa.hibernate.ddl-auto=update
|
spring.jpa.hibernate.ddl-auto=update
|
||||||
@@ -281,20 +281,35 @@ The output contains
|
|||||||
|
|
||||||
Add these patches to the kustomization:
|
Add these patches to the kustomization:
|
||||||
|
|
||||||
<!-- @addPatch @testAgainstLatestRelease -->
|
<!-- @addPatch -->
|
||||||
```
|
```
|
||||||
cd $DEMO_HOME
|
cd $DEMO_HOME
|
||||||
kustomize edit add patch memorylimit_patch.yaml
|
kustomize edit add patch --path memorylimit_patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
kustomize edit add patch healthcheck_patch.yaml
|
kustomize edit add patch --path healthcheck_patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
```
|
```
|
||||||
|
|
||||||
`kustomization.yaml` should have patches field:
|
`kustomization.yaml` should have patches field:
|
||||||
|
|
||||||
> ```
|
> ```
|
||||||
> patchesStrategicMerge:
|
> patches:
|
||||||
> - patch.yaml
|
> - path: patch.yaml
|
||||||
> - memorylimit_patch.yaml
|
> target:
|
||||||
> - healthcheck_patch.yaml
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
|
> - path: memorylimit_patch.yaml
|
||||||
|
> target:
|
||||||
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
|
> - path: healthcheck_patch.yaml
|
||||||
|
> target:
|
||||||
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
> ```
|
> ```
|
||||||
|
|
||||||
The output of the following command can now be applied
|
The output of the following command can now be applied
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
## ConfigMap 的生成和滚动更新
|
## ConfigMap 的生成和滚动更新
|
||||||
|
|
||||||
kustomize 提供了两种添加 ConfigMap 的方法:
|
kustomize 提供了两种添加 ConfigMap 的方法:
|
||||||
|
|
||||||
- 将 ConfigMap 声明为 [resource]
|
- 将 ConfigMap 声明为 [resource]
|
||||||
- 通过 ConfigMapGenerator 声明 ConfigMap
|
- 通过 ConfigMapGenerator 声明 ConfigMap
|
||||||
|
|
||||||
@@ -14,7 +15,7 @@ kustomize 提供了两种添加 ConfigMap 的方法:
|
|||||||
> # 将 ConfigMap 声明为 resource
|
> # 将 ConfigMap 声明为 resource
|
||||||
> resources:
|
> resources:
|
||||||
> - configmap.yaml
|
> - configmap.yaml
|
||||||
>
|
>
|
||||||
> # 在 ConfigMapGenerator 中声明 ConfigMap
|
> # 在 ConfigMapGenerator 中声明 ConfigMap
|
||||||
> configMapGenerator:
|
> configMapGenerator:
|
||||||
> - name: a-configmap
|
> - name: a-configmap
|
||||||
@@ -30,7 +31,9 @@ kustomize 提供了两种添加 ConfigMap 的方法:
|
|||||||
### 建立 base 和 staging
|
### 建立 base 和 staging
|
||||||
|
|
||||||
使用 configMapGenerator 建立 base
|
使用 configMapGenerator 建立 base
|
||||||
|
|
||||||
<!-- @establishBase @testAgainstLatestRelease -->
|
<!-- @establishBase @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
DEMO_HOME=$(mktemp -d)
|
DEMO_HOME=$(mktemp -d)
|
||||||
|
|
||||||
@@ -48,16 +51,18 @@ commonLabels:
|
|||||||
resources:
|
resources:
|
||||||
- deployment.yaml
|
- deployment.yaml
|
||||||
- service.yaml
|
- service.yaml
|
||||||
configMapGenerator:
|
configMapGenerator:
|
||||||
- name: the-map
|
- name: the-map
|
||||||
literals:
|
literals:
|
||||||
- altGreeting=Good Morning!
|
- altGreeting=Good Morning!
|
||||||
- enableRisky="false"
|
- enableRisky="false"
|
||||||
EOF
|
EOF
|
||||||
```
|
```
|
||||||
|
|
||||||
通过应用 ConfigMap patch 的方式建立 staging
|
通过应用 ConfigMap patch 的方式建立 staging
|
||||||
|
|
||||||
<!-- @establishStaging @testAgainstLatestRelease -->
|
<!-- @establishStaging @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
OVERLAYS=$DEMO_HOME/overlays
|
OVERLAYS=$DEMO_HOME/overlays
|
||||||
mkdir -p $OVERLAYS/staging
|
mkdir -p $OVERLAYS/staging
|
||||||
@@ -94,6 +99,7 @@ EOF
|
|||||||
deployment 按照名称引用此 ConfigMap :
|
deployment 按照名称引用此 ConfigMap :
|
||||||
|
|
||||||
<!-- @showDeployment @testAgainstLatestRelease -->
|
<!-- @showDeployment @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
grep -C 2 configMapKeyRef $BASE/deployment.yaml
|
grep -C 2 configMapKeyRef $BASE/deployment.yaml
|
||||||
```
|
```
|
||||||
@@ -102,16 +108,17 @@ grep -C 2 configMapKeyRef $BASE/deployment.yaml
|
|||||||
|
|
||||||
更改 Deployment 配置的推荐方法是:
|
更改 Deployment 配置的推荐方法是:
|
||||||
|
|
||||||
1. 使用新名称创建一个新的 configMap
|
1. 使用新名称创建一个新的 configMap
|
||||||
2. 为_deployment_ 添加 patch,修改相应 `configMapKeyRef` 字段的名称值。
|
2. 为*deployment* 添加 patch,修改相应 `configMapKeyRef` 字段的名称值。
|
||||||
|
|
||||||
后一种更改会启动对 deployment 中的 pod 的滚动更新。旧的 configMap 在不再被任何其他资源引用时最终会被[垃圾回收](/../../issues/242)。
|
后一种更改会启动对 deployment 中的 pod 的滚动更新。旧的 configMap 在不再被任何其他资源引用时最终会被[垃圾回收](/../../issues/242)。
|
||||||
|
|
||||||
### 如何使用 kustomize
|
### 如何使用 kustomize
|
||||||
|
|
||||||
_staging_ 的 [variant] 包含一个 configMap 的 [patch]:
|
_staging_ 的 [variant] 包含一个 configMap 的 [patch]:
|
||||||
|
|
||||||
<!-- @showMapPatch @testAgainstLatestRelease -->
|
<!-- @showMapPatch @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
cat $OVERLAYS/staging/map.yaml
|
cat $OVERLAYS/staging/map.yaml
|
||||||
```
|
```
|
||||||
@@ -121,6 +128,7 @@ cat $OVERLAYS/staging/map.yaml
|
|||||||
在 ConfigMapGenerator 中声明 ConfigMap 的修改。
|
在 ConfigMapGenerator 中声明 ConfigMap 的修改。
|
||||||
|
|
||||||
<!-- @showMapBase @testAgainstLatestRelease -->
|
<!-- @showMapBase @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
grep -C 4 configMapGenerator $BASE/kustomization.yaml
|
grep -C 4 configMapGenerator $BASE/kustomization.yaml
|
||||||
```
|
```
|
||||||
@@ -130,6 +138,7 @@ grep -C 4 configMapGenerator $BASE/kustomization.yaml
|
|||||||
但是,文件中指定的名称值不是群集中使用的名称值。根据设计,kustomize 修改从 ConfigMapGenerator 声明的 ConfigMaps 的名称。要查看最终在群集中使用的名称,只需运行 kustomize:
|
但是,文件中指定的名称值不是群集中使用的名称值。根据设计,kustomize 修改从 ConfigMapGenerator 声明的 ConfigMaps 的名称。要查看最终在群集中使用的名称,只需运行 kustomize:
|
||||||
|
|
||||||
<!-- @grepStagingName @testAgainstLatestRelease -->
|
<!-- @grepStagingName @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging |\
|
kustomize build $OVERLAYS/staging |\
|
||||||
grep -B 8 -A 1 staging-the-map
|
grep -B 8 -A 1 staging-the-map
|
||||||
@@ -141,7 +150,8 @@ kustomize build $OVERLAYS/staging |\
|
|||||||
|
|
||||||
configMap 名称的后缀是由 map 内容的哈希生成的 - 在这种情况下,名称后缀是 _5276h4th55_ :
|
configMap 名称的后缀是由 map 内容的哈希生成的 - 在这种情况下,名称后缀是 _5276h4th55_ :
|
||||||
|
|
||||||
<!-- @grepStagingHash -->
|
<!-- @grepStagingHash @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging | grep 5276h4th55
|
kustomize build $OVERLAYS/staging | grep 5276h4th55
|
||||||
```
|
```
|
||||||
@@ -149,6 +159,7 @@ kustomize build $OVERLAYS/staging | grep 5276h4th55
|
|||||||
现在修改 map patch ,更改该服务将使用的问候消息:
|
现在修改 map patch ,更改该服务将使用的问候消息:
|
||||||
|
|
||||||
<!-- @changeMap @testAgainstLatestRelease -->
|
<!-- @changeMap @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
sed -i.bak 's/pineapple/kiwi/' $OVERLAYS/staging/map.yaml
|
sed -i.bak 's/pineapple/kiwi/' $OVERLAYS/staging/map.yaml
|
||||||
```
|
```
|
||||||
@@ -163,6 +174,7 @@ kustomize build $OVERLAYS/staging |\
|
|||||||
再次运行 kustomize 查看新的 configMap 名称:
|
再次运行 kustomize 查看新的 configMap 名称:
|
||||||
|
|
||||||
<!-- @grepStagingName @testAgainstLatestRelease -->
|
<!-- @grepStagingName @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
kustomize build $OVERLAYS/staging |\
|
kustomize build $OVERLAYS/staging |\
|
||||||
grep -B 8 -A 1 staging-the-map
|
grep -B 8 -A 1 staging-the-map
|
||||||
@@ -170,7 +182,8 @@ kustomize build $OVERLAYS/staging |\
|
|||||||
|
|
||||||
确认 configMap 内容的更改将会生成以 _c2g8fcbf88_ 结尾的三个新名称 - 一个在 configMap 的名称中,另两个在使用 ConfigMap 的 deployment 中:
|
确认 configMap 内容的更改将会生成以 _c2g8fcbf88_ 结尾的三个新名称 - 一个在 configMap 的名称中,另两个在使用 ConfigMap 的 deployment 中:
|
||||||
|
|
||||||
<!-- @countHashes -->
|
<!-- @countHashes @testAgainstLatestRelease -->
|
||||||
|
|
||||||
```
|
```
|
||||||
test 3 == \
|
test 3 == \
|
||||||
$(kustomize build $OVERLAYS/staging | grep c2g8fcbf88 | wc -l); \
|
$(kustomize build $OVERLAYS/staging | grep c2g8fcbf88 | wc -l); \
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ cat kustomization.yaml
|
|||||||
1. 通过 patch 添加一个环境变量
|
1. 通过 patch 添加一个环境变量
|
||||||
2. 将文件添加到 ConfigMap 中
|
2. 将文件添加到 ConfigMap 中
|
||||||
|
|
||||||
<!-- @customizeConfigMap @testAgainstLatestRelease -->
|
<!-- @customizeConfigMap -->
|
||||||
```
|
```
|
||||||
cat <<EOF >$DEMO_HOME/patch.yaml
|
cat <<EOF >$DEMO_HOME/patch.yaml
|
||||||
apiVersion: apps/v1
|
apiVersion: apps/v1
|
||||||
@@ -106,7 +106,7 @@ spec:
|
|||||||
value: prod
|
value: prod
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
kustomize edit add patch patch.yaml
|
kustomize edit add patch --path patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
|
|
||||||
cat <<EOF >$DEMO_HOME/application-prod.properties
|
cat <<EOF >$DEMO_HOME/application-prod.properties
|
||||||
spring.jpa.hibernate.ddl-auto=update
|
spring.jpa.hibernate.ddl-auto=update
|
||||||
@@ -257,23 +257,38 @@ cat $DEMO_HOME/healthcheck_patch.yaml
|
|||||||
|
|
||||||
将这些 patch 添加到 `kustomization.yaml` 中:
|
将这些 patch 添加到 `kustomization.yaml` 中:
|
||||||
|
|
||||||
<!-- @addPatch @testAgainstLatestRelease -->
|
<!-- @addPatch -->
|
||||||
```
|
```
|
||||||
cd $DEMO_HOME
|
cd $DEMO_HOME
|
||||||
kustomize edit add patch memorylimit_patch.yaml
|
kustomize edit add patch --path memorylimit_patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
kustomize edit add patch healthcheck_patch.yaml
|
kustomize edit add patch --path healthcheck_patch.yaml --name sbdemo --kind Deployment --group apps --version v1
|
||||||
```
|
```
|
||||||
|
|
||||||
执行上面的命令后,`kustomization.yaml` 的 patchesStrategicMerge 字段如下:
|
执行上面的命令后,`kustomization.yaml` 的 patches 字段如下:
|
||||||
|
|
||||||
> ```
|
> ```
|
||||||
> patchesStrategicMerge:
|
> patches:
|
||||||
> - patch.yaml
|
> - path: patch.yaml
|
||||||
> - memorylimit_patch.yaml
|
> target:
|
||||||
> - healthcheck_patch.yaml
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
|
> - path: memorylimit_patch.yaml
|
||||||
|
> target:
|
||||||
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
|
> - path: healthcheck_patch.yaml
|
||||||
|
> target:
|
||||||
|
> group: apps
|
||||||
|
> version: v1
|
||||||
|
> kind: Deployment
|
||||||
|
> name: sbdemo
|
||||||
> ```
|
> ```
|
||||||
|
|
||||||
现在就可以将完整的配置输出并在集群中部署(将结果通过管道输出给 `kubectl apply`),在生产环境创建Spring Boot 应用。
|
现在就可以将完整的配置输出并在集群中部署(将结果通过管道输出给 `kubectl apply`),在生产环境创建 Spring Boot 应用。
|
||||||
|
|
||||||
<!-- @finalBuild @testAgainstLatestRelease -->
|
<!-- @finalBuild @testAgainstLatestRelease -->
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -9,54 +9,67 @@ import (
|
|||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/edit/patch"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/kustfile"
|
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/kustfile"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/util"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type addPatchOptions struct {
|
type addPatchOptions struct {
|
||||||
patchFilePaths []string
|
Patch types.Patch
|
||||||
}
|
}
|
||||||
|
|
||||||
// newCmdAddPatch adds the name of a file containing a patch to the kustomization file.
|
// newCmdAddPatch adds the name of a file containing a patch to the kustomization file.
|
||||||
func newCmdAddPatch(fSys filesys.FileSystem) *cobra.Command {
|
func newCmdAddPatch(fSys filesys.FileSystem) *cobra.Command {
|
||||||
var o addPatchOptions
|
var o addPatchOptions
|
||||||
|
o.Patch.Target = &types.Selector{}
|
||||||
|
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "patch",
|
Use: "patch",
|
||||||
Short: "Add the name of a file containing a patch to the kustomization file.",
|
Short: "Add an item to patches field.",
|
||||||
|
Long: `This command will add an item to patches field in the kustomization file.
|
||||||
|
Each item may:
|
||||||
|
|
||||||
|
- be either a strategic merge patch, or a JSON patch
|
||||||
|
- be either a file, or an inline string
|
||||||
|
- target a single resource or multiple resources
|
||||||
|
|
||||||
|
For more information please see https://kubernetes-sigs.github.io/kustomize/api-reference/kustomization/patches/
|
||||||
|
`,
|
||||||
Example: `
|
Example: `
|
||||||
add patch {filepath}`,
|
add patch --path {filepath} --group {target group name} --version {target version}`,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
err := o.Validate(args)
|
err := o.Validate()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return o.RunAddPatch(fSys)
|
return o.RunAddPatch(fSys)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Path, "path", "", "Path to the patch file. Cannot be used with --patch at the same time.")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Patch, "patch", "", "Literal string of patch content. Cannot be used with --path at the same time.")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Group, "group", "", "API group in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Version, "version", "", "API version in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Kind, "kind", "", "Resource kind in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Name, "name", "", "Resource name in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Namespace, "namespace", "", "Resource namespace in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.AnnotationSelector, "annotation-selector", "", "annotationSelector in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.LabelSelector, "label-selector", "", "labelSelector in patch target")
|
||||||
|
|
||||||
return cmd
|
return cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate validates addPatch command.
|
// Validate validates addPatch command.
|
||||||
func (o *addPatchOptions) Validate(args []string) error {
|
func (o *addPatchOptions) Validate() error {
|
||||||
if len(args) == 0 {
|
if o.Patch.Patch != "" && o.Patch.Path != "" {
|
||||||
return errors.New("must specify a patch file")
|
return errors.New("patch and path can't be set at the same time")
|
||||||
|
}
|
||||||
|
if o.Patch.Patch == "" && o.Patch.Path == "" {
|
||||||
|
return errors.New("must provide either patch or path")
|
||||||
}
|
}
|
||||||
o.patchFilePaths = args
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// RunAddPatch runs addPatch command (do real work).
|
// RunAddPatch runs addPatch command (do real work).
|
||||||
func (o *addPatchOptions) RunAddPatch(fSys filesys.FileSystem) error {
|
func (o *addPatchOptions) RunAddPatch(fSys filesys.FileSystem) error {
|
||||||
patches, err := util.GlobPatterns(fSys, o.patchFilePaths)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(patches) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
mf, err := kustfile.NewKustomizationFile(fSys)
|
mf, err := kustfile.NewKustomizationFile(fSys)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -67,13 +80,18 @@ func (o *addPatchOptions) RunAddPatch(fSys filesys.FileSystem) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, p := range patches {
|
// Omit target if it's empty
|
||||||
if patch.Exist(m.PatchesStrategicMerge, p) {
|
emptyTarget := types.Selector{}
|
||||||
log.Printf("patch %s already in kustomization file", p)
|
if o.Patch.Target != nil && *o.Patch.Target == emptyTarget {
|
||||||
continue
|
o.Patch.Target = nil
|
||||||
}
|
|
||||||
m.PatchesStrategicMerge = patch.Append(m.PatchesStrategicMerge, p)
|
|
||||||
}
|
}
|
||||||
|
for _, p := range m.Patches {
|
||||||
|
if p.Equals(o.Patch) {
|
||||||
|
log.Printf("patch %#v already in kustomization file", p)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m.Patches = append(m.Patches, o.Patch)
|
||||||
|
|
||||||
return mf.Write(m)
|
return mf.Write(m)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,19 +15,34 @@ const (
|
|||||||
patchFileName = "myWonderfulPatch.yaml"
|
patchFileName = "myWonderfulPatch.yaml"
|
||||||
patchFileContent = `
|
patchFileContent = `
|
||||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit,
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit,
|
||||||
sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.
|
|
||||||
`
|
`
|
||||||
|
kind = "myKind"
|
||||||
|
group = "myGroup"
|
||||||
|
version = "myVersion"
|
||||||
|
name = "myName"
|
||||||
|
namespace = "myNamespace"
|
||||||
|
annotationSelector = "myAnnotationSelector"
|
||||||
|
labelSelector = "myLabelSelector"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestAddPatchHappyPath(t *testing.T) {
|
func TestAddPatchWithFilePath(t *testing.T) {
|
||||||
fSys := filesys.MakeEmptyDirInMemory()
|
fSys := filesys.MakeEmptyDirInMemory()
|
||||||
fSys.WriteFile(patchFileName, []byte(patchFileContent))
|
fSys.WriteFile(patchFileName, []byte(patchFileContent))
|
||||||
fSys.WriteFile(patchFileName+"another", []byte(patchFileContent))
|
|
||||||
testutils_test.WriteTestKustomization(fSys)
|
testutils_test.WriteTestKustomization(fSys)
|
||||||
|
|
||||||
cmd := newCmdAddPatch(fSys)
|
cmd := newCmdAddPatch(fSys)
|
||||||
args := []string{patchFileName + "*"}
|
args := []string{
|
||||||
err := cmd.RunE(cmd, args)
|
"--path", patchFileName,
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected cmd error: %v", err)
|
t.Errorf("unexpected cmd error: %v", err)
|
||||||
}
|
}
|
||||||
@@ -35,11 +50,42 @@ func TestAddPatchHappyPath(t *testing.T) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected read error: %v", err)
|
t.Errorf("unexpected read error: %v", err)
|
||||||
}
|
}
|
||||||
if !strings.Contains(string(content), patchFileName) {
|
for i := 1; i < len(args); i += 2 {
|
||||||
t.Errorf("expected patch name in kustomization")
|
if !strings.Contains(string(content), args[i]) {
|
||||||
|
t.Errorf("expected flag value of %s in kustomization but got\n%s", args[i-1], content)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if !strings.Contains(string(content), patchFileName+"another") {
|
}
|
||||||
t.Errorf("expected patch name in kustomization")
|
|
||||||
|
func TestAddPatchWithPatchContent(t *testing.T) {
|
||||||
|
fSys := filesys.MakeEmptyDirInMemory()
|
||||||
|
fSys.WriteFile(patchFileName, []byte(patchFileContent))
|
||||||
|
testutils_test.WriteTestKustomization(fSys)
|
||||||
|
|
||||||
|
cmd := newCmdAddPatch(fSys)
|
||||||
|
args := []string{
|
||||||
|
"--patch", patchFileContent,
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("unexpected cmd error: %v", err)
|
||||||
|
}
|
||||||
|
content, err := testutils_test.ReadTestKustomization(fSys)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("unexpected read error: %v", err)
|
||||||
|
}
|
||||||
|
for i := 1; i < len(args); i += 2 {
|
||||||
|
if !strings.Contains(string(content), strings.Trim(args[i], " \n")) {
|
||||||
|
t.Errorf("expected flag value of %s in kustomization but got\n%s", args[i-1], content)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -49,14 +95,24 @@ func TestAddPatchAlreadyThere(t *testing.T) {
|
|||||||
testutils_test.WriteTestKustomization(fSys)
|
testutils_test.WriteTestKustomization(fSys)
|
||||||
|
|
||||||
cmd := newCmdAddPatch(fSys)
|
cmd := newCmdAddPatch(fSys)
|
||||||
args := []string{patchFileName}
|
args := []string{
|
||||||
err := cmd.RunE(cmd, args)
|
"--path", patchFileName,
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected cmd error: %v", err)
|
t.Fatalf("unexpected cmd error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// adding an existing patch shouldn't return an error
|
// adding an existing patch shouldn't return an error
|
||||||
err = cmd.RunE(cmd, args)
|
err = cmd.Execute()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected cmd error: %v", err)
|
t.Errorf("unexpected cmd error: %v", err)
|
||||||
}
|
}
|
||||||
@@ -70,7 +126,7 @@ func TestAddPatchNoArgs(t *testing.T) {
|
|||||||
if err == nil {
|
if err == nil {
|
||||||
t.Errorf("expected error: %v", err)
|
t.Errorf("expected error: %v", err)
|
||||||
}
|
}
|
||||||
if err.Error() != "must specify a patch file" {
|
if err.Error() != "must provide either patch or path" {
|
||||||
t.Errorf("incorrect error: %v", err.Error())
|
t.Errorf("incorrect error: %v", err.Error())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ func NewCmdAdd(
|
|||||||
kustomize edit add resource <filepath>
|
kustomize edit add resource <filepath>
|
||||||
|
|
||||||
# Adds a patch to the kustomization
|
# Adds a patch to the kustomization
|
||||||
kustomize edit add patch <filepath>
|
kustomize edit add patch --path {filepath} --group {target group name} --version {target version}
|
||||||
|
|
||||||
# Adds a component to the kustomization
|
# Adds a component to the kustomization
|
||||||
kustomize edit add component <filepath>
|
kustomize edit add component <filepath>
|
||||||
|
|||||||
@@ -1,41 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
package patch
|
|
||||||
|
|
||||||
import "sigs.k8s.io/kustomize/api/types"
|
|
||||||
|
|
||||||
// Append appends a slice of patch paths to a PatchStrategicMerge slice
|
|
||||||
func Append(patches []types.PatchStrategicMerge, paths ...string) []types.PatchStrategicMerge {
|
|
||||||
for _, p := range paths {
|
|
||||||
patches = append(patches, types.PatchStrategicMerge(p))
|
|
||||||
}
|
|
||||||
return patches
|
|
||||||
}
|
|
||||||
|
|
||||||
// Exist determines if a patch path exists in a slice of PatchStrategicMerge
|
|
||||||
func Exist(patches []types.PatchStrategicMerge, path string) bool {
|
|
||||||
for _, p := range patches {
|
|
||||||
if p == types.PatchStrategicMerge(path) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete deletes patches from a PatchStrategicMerge slice
|
|
||||||
func Delete(patches []types.PatchStrategicMerge, paths ...string) []types.PatchStrategicMerge {
|
|
||||||
// Convert paths into PatchStrategicMerge slice
|
|
||||||
convertedPath := make([]types.PatchStrategicMerge, len(paths))
|
|
||||||
for i, p := range paths {
|
|
||||||
convertedPath[i] = types.PatchStrategicMerge(p)
|
|
||||||
}
|
|
||||||
|
|
||||||
filteredPatches := make([]types.PatchStrategicMerge, 0, len(patches))
|
|
||||||
for _, containedPatch := range patches {
|
|
||||||
if !Exist(convertedPath, string(containedPatch)) {
|
|
||||||
filteredPatches = append(filteredPatches, containedPatch)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return filteredPatches
|
|
||||||
}
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
// Copyright 2019 The Kubernetes Authors.
|
|
||||||
// SPDX-License-Identifier: Apache-2.
|
|
||||||
|
|
||||||
package patch
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
func buildPatchStrategicMergeSlice(patchStrings []string) []types.PatchStrategicMerge {
|
|
||||||
var patches []types.PatchStrategicMerge
|
|
||||||
for _, patchString := range patchStrings {
|
|
||||||
patches = append(patches, types.PatchStrategicMerge(patchString))
|
|
||||||
}
|
|
||||||
return patches
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAppend(t *testing.T) {
|
|
||||||
patchStrings := []string{"patch1.yaml", "patch2.yaml"}
|
|
||||||
patches := buildPatchStrategicMergeSlice(patchStrings)
|
|
||||||
|
|
||||||
patches = Append(patches, "patch3.yaml")
|
|
||||||
|
|
||||||
for i, k := range []string{"patch1.yaml", "patch2.yaml", "patch3.yaml"} {
|
|
||||||
if patches[i] != types.PatchStrategicMerge(k) {
|
|
||||||
t.Fatalf("patches[%d] must be %s, got %s", i, k, patches[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestExistTrue(t *testing.T) {
|
|
||||||
patchStrings := []string{"patch1.yaml", "patch2.yaml"}
|
|
||||||
patches := buildPatchStrategicMergeSlice(patchStrings)
|
|
||||||
|
|
||||||
for _, patchString := range patchStrings {
|
|
||||||
if !Exist(patches, patchString) {
|
|
||||||
t.Fatalf("%s must exist", patchString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestExistFalse(t *testing.T) {
|
|
||||||
patchStrings := []string{"patch1.yaml", "patch2.yaml"}
|
|
||||||
patches := buildPatchStrategicMergeSlice(patchStrings)
|
|
||||||
|
|
||||||
for _, patchString := range []string{"invalid1.yaml", "invalid2.yaml"} {
|
|
||||||
if Exist(patches, patchString) {
|
|
||||||
t.Fatalf("%s must not exist", patchString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDelete(t *testing.T) {
|
|
||||||
patchStrings := []string{"patch1.yaml", "patch2.yaml"}
|
|
||||||
patches := buildPatchStrategicMergeSlice(patchStrings)
|
|
||||||
|
|
||||||
patches = Delete(patches, "patch1.yaml")
|
|
||||||
|
|
||||||
if Exist(patches, "patch1.yaml") {
|
|
||||||
t.Fatalf("patch1.yaml should be deleted")
|
|
||||||
}
|
|
||||||
if !Exist(patches, "patch2.yaml") {
|
|
||||||
t.Fatalf("patch2.yaml should exist")
|
|
||||||
}
|
|
||||||
if len(patches) != 1 {
|
|
||||||
t.Fatalf("Length of slice must be 1: actual %d", len(patches))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDeleteMultiple(t *testing.T) {
|
|
||||||
patchStrings := []string{"patch1.yaml", "patch2.yaml"}
|
|
||||||
patches := buildPatchStrategicMergeSlice(patchStrings)
|
|
||||||
|
|
||||||
patches = Delete(patches, "patch2.yaml", "patch4.yaml", "patch1.yaml", "patch3.yaml")
|
|
||||||
|
|
||||||
for _, k := range patchStrings {
|
|
||||||
if Exist(patches, k) {
|
|
||||||
t.Fatalf("%s should be deleted", k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -23,7 +23,7 @@ func NewCmdRemove(
|
|||||||
kustomize edit remove resource {pattern}
|
kustomize edit remove resource {pattern}
|
||||||
|
|
||||||
# Removes one or more patches from the kustomization file
|
# Removes one or more patches from the kustomization file
|
||||||
kustomize edit remove patch <filepath>
|
kustomize edit remove patch --path {filepath} --group {target group name} --version {target version}
|
||||||
|
|
||||||
# Removes one or more commonLabels from the kustomization file
|
# Removes one or more commonLabels from the kustomization file
|
||||||
kustomize edit remove label {labelKey1},{labelKey2}
|
kustomize edit remove label {labelKey1},{labelKey2}
|
||||||
|
|||||||
@@ -10,55 +10,58 @@ import (
|
|||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/api/konfig"
|
"sigs.k8s.io/kustomize/api/konfig"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/edit/patch"
|
"sigs.k8s.io/kustomize/api/types"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/kustfile"
|
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/kustfile"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/util"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type removePatchOptions struct {
|
type removePatchOptions struct {
|
||||||
patchFilePaths []string
|
Patch types.Patch
|
||||||
}
|
}
|
||||||
|
|
||||||
// newCmdRemovePatch removes the name of a file containing a patch from the kustomization file.
|
// newCmdRemovePatch removes the name of a file containing a patch from the kustomization file.
|
||||||
func newCmdRemovePatch(fSys filesys.FileSystem) *cobra.Command {
|
func newCmdRemovePatch(fSys filesys.FileSystem) *cobra.Command {
|
||||||
var o removePatchOptions
|
var o removePatchOptions
|
||||||
|
o.Patch.Target = &types.Selector{}
|
||||||
|
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "patch",
|
Use: "patch",
|
||||||
Short: "Removes one or more patches from " +
|
Short: "Removes a patch from " +
|
||||||
konfig.DefaultKustomizationFileName(),
|
konfig.DefaultKustomizationFileName(),
|
||||||
|
Long: `Removes a patch from patches field. The fields specified by flags must
|
||||||
|
exactly match the patch item to successfully remote the item.`,
|
||||||
Example: `
|
Example: `
|
||||||
remove patch {filepath}`,
|
remove patch --path {filepath} --group {target group name} --version {target version}`,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
err := o.Validate(args)
|
err := o.Validate()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return o.RunRemovePatch(fSys)
|
return o.RunRemovePatch(fSys)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Path, "path", "", "Path to the patch file. Cannot be used with --patch at the same time.")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Patch, "patch", "", "Literal string of patch content. Cannot be used with --path at the same time.")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Group, "group", "", "API group in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Version, "version", "", "API version in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Kind, "kind", "", "Resource kind in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Name, "name", "", "Resource name in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.Namespace, "namespace", "", "Resource namespace in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.AnnotationSelector, "annotation-selector", "", "annotationSelector in patch target")
|
||||||
|
cmd.Flags().StringVar(&o.Patch.Target.LabelSelector, "label-selector", "", "labelSelector in patch target")
|
||||||
|
|
||||||
return cmd
|
return cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate validates removePatch command.
|
// Validate validates removePatch command.
|
||||||
func (o *removePatchOptions) Validate(args []string) error {
|
func (o *removePatchOptions) Validate() error {
|
||||||
if len(args) == 0 {
|
if o.Patch.Patch != "" && o.Patch.Path != "" {
|
||||||
return errors.New("must specify a patch file")
|
return errors.New("patch and path can't be set at the same time")
|
||||||
}
|
}
|
||||||
o.patchFilePaths = args
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// RunRemovePatch runs removePatch command (do real work).
|
// RunRemovePatch runs removePatch command (do real work).
|
||||||
func (o *removePatchOptions) RunRemovePatch(fSys filesys.FileSystem) error {
|
func (o *removePatchOptions) RunRemovePatch(fSys filesys.FileSystem) error {
|
||||||
patches, err := util.GlobPatterns(fSys, o.patchFilePaths)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if len(patches) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
mf, err := kustfile.NewKustomizationFile(fSys)
|
mf, err := kustfile.NewKustomizationFile(fSys)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@@ -69,15 +72,23 @@ func (o *removePatchOptions) RunRemovePatch(fSys filesys.FileSystem) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
var removePatches []string
|
// Omit target if it's empty
|
||||||
for _, p := range patches {
|
emptyTarget := types.Selector{}
|
||||||
if !patch.Exist(m.PatchesStrategicMerge, p) {
|
if o.Patch.Target != nil && *o.Patch.Target == emptyTarget {
|
||||||
log.Printf("patch %s doesn't exist in kustomization file", p)
|
o.Patch.Target = nil
|
||||||
continue
|
|
||||||
}
|
|
||||||
removePatches = append(removePatches, p)
|
|
||||||
}
|
}
|
||||||
m.PatchesStrategicMerge = patch.Delete(m.PatchesStrategicMerge, removePatches...)
|
|
||||||
|
var patches []types.Patch
|
||||||
|
for _, p := range m.Patches {
|
||||||
|
if !p.Equals(o.Patch) {
|
||||||
|
patches = append(patches, p)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(patches) == len(m.Patches) {
|
||||||
|
log.Printf("patch %s doesn't exist in kustomization file", o.Patch)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
m.Patches = patches
|
||||||
|
|
||||||
return mf.Write(m)
|
return mf.Write(m)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,29 +4,53 @@
|
|||||||
package remove
|
package remove
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"sigs.k8s.io/kustomize/api/filesys"
|
"sigs.k8s.io/kustomize/api/filesys"
|
||||||
"sigs.k8s.io/kustomize/kustomize/v3/internal/commands/edit/patch"
|
|
||||||
testutils_test "sigs.k8s.io/kustomize/kustomize/v3/internal/commands/testutils"
|
testutils_test "sigs.k8s.io/kustomize/kustomize/v3/internal/commands/testutils"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
patchFileContent = `
|
patchFileContent = `- op: replace
|
||||||
Lorem ipsum dolor sit amet, consectetur adipiscing elit,
|
path: /some/existing/path
|
||||||
sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.
|
value: new value`
|
||||||
`
|
kind = "myKind"
|
||||||
|
group = "myGroup"
|
||||||
|
version = "myVersion"
|
||||||
|
name = "myName"
|
||||||
|
namespace = "myNamespace"
|
||||||
|
annotationSelector = "myAnnotationSelector"
|
||||||
|
labelSelector = "myLabelSelector"
|
||||||
)
|
)
|
||||||
|
|
||||||
func makeKustomizationPatchFS() filesys.FileSystem {
|
func makeKustomizationPatchFS() filesys.FileSystem {
|
||||||
fSys := filesys.MakeEmptyDirInMemory()
|
fSys := filesys.MakeEmptyDirInMemory()
|
||||||
patches := []string{"patch1.yaml", "patch2.yaml"}
|
patches := []string{"patch1.yaml", "patch2.yaml"}
|
||||||
|
|
||||||
testutils_test.WriteTestKustomizationWith(fSys, []byte(
|
testutils_test.WriteTestKustomizationWith(fSys, []byte(`
|
||||||
fmt.Sprintf("patchesStrategicMerge:\n - %s",
|
patches:
|
||||||
strings.Join(patches, "\n - "))))
|
- path: patch1.yaml
|
||||||
|
target:
|
||||||
|
group: myGroup
|
||||||
|
version: myVersion
|
||||||
|
kind: myKind
|
||||||
|
name: myName
|
||||||
|
namespace: myNamespace
|
||||||
|
labelSelector: myLabelSelector
|
||||||
|
annotationSelector: myAnnotationSelector
|
||||||
|
- path: patch2.yaml
|
||||||
|
target:
|
||||||
|
group: myGroup
|
||||||
|
version: myVersion
|
||||||
|
kind: myKind
|
||||||
|
- patch: |-
|
||||||
|
- op: replace
|
||||||
|
path: /some/existing/path
|
||||||
|
value: new value
|
||||||
|
target:
|
||||||
|
kind: myKind
|
||||||
|
labelSelector: myLabelSelector
|
||||||
|
`))
|
||||||
|
|
||||||
for _, p := range patches {
|
for _, p := range patches {
|
||||||
fSys.WriteFile(p, []byte(patchFileContent))
|
fSys.WriteFile(p, []byte(patchFileContent))
|
||||||
@@ -38,69 +62,86 @@ func makeKustomizationPatchFS() filesys.FileSystem {
|
|||||||
func TestRemovePatch(t *testing.T) {
|
func TestRemovePatch(t *testing.T) {
|
||||||
fSys := makeKustomizationPatchFS()
|
fSys := makeKustomizationPatchFS()
|
||||||
cmd := newCmdRemovePatch(fSys)
|
cmd := newCmdRemovePatch(fSys)
|
||||||
args := []string{"patch1.yaml"}
|
patchPath := "patch1.yaml"
|
||||||
err := cmd.RunE(cmd, args)
|
args := []string{
|
||||||
|
"--path", patchPath,
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected error %v", err)
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m := readKustomizationFS(t, fSys)
|
m := readKustomizationFS(t, fSys)
|
||||||
for _, k := range args {
|
for _, p := range m.Patches {
|
||||||
if patch.Exist(m.PatchesStrategicMerge, k) {
|
if p.Path == patchPath {
|
||||||
t.Errorf("%s must be deleted", k)
|
t.Fatalf("%s must be deleted", patchPath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRemovePatchMultipleArgs(t *testing.T) {
|
func TestRemovePatch2(t *testing.T) {
|
||||||
fSys := makeKustomizationPatchFS()
|
fSys := makeKustomizationPatchFS()
|
||||||
cmd := newCmdRemovePatch(fSys)
|
cmd := newCmdRemovePatch(fSys)
|
||||||
args := []string{"patch1.yaml", "patch2.yaml"}
|
args := []string{
|
||||||
err := cmd.RunE(cmd, args)
|
"--patch", patchFileContent,
|
||||||
|
"--kind", kind,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected error %v", err)
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m := readKustomizationFS(t, fSys)
|
m := readKustomizationFS(t, fSys)
|
||||||
for _, k := range args {
|
for _, p := range m.Patches {
|
||||||
if patch.Exist(m.PatchesStrategicMerge, k) {
|
if p.Patch == patchFileContent {
|
||||||
t.Errorf("%s must be deleted", k)
|
t.Fatalf("%s must be deleted", patchFileContent)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRemovePatchGlob(t *testing.T) {
|
|
||||||
fSys := makeKustomizationPatchFS()
|
|
||||||
cmd := newCmdRemovePatch(fSys)
|
|
||||||
args := []string{"patch*.yaml"}
|
|
||||||
err := cmd.RunE(cmd, args)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
t.Errorf("unexpected error %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
m := readKustomizationFS(t, fSys)
|
|
||||||
if len(m.PatchesStrategicMerge) != 0 {
|
|
||||||
t.Errorf("all patch must be deleted")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRemovePatchNotDefinedInKustomization(t *testing.T) {
|
func TestRemovePatchNotDefinedInKustomization(t *testing.T) {
|
||||||
fSys := makeKustomizationPatchFS()
|
fSys := makeKustomizationPatchFS()
|
||||||
cmd := newCmdRemovePatch(fSys)
|
cmd := newCmdRemovePatch(fSys)
|
||||||
args := []string{"patch3.yaml"}
|
args := []string{
|
||||||
err := cmd.RunE(cmd, args)
|
"--path", "patch3.yaml",
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected error %v", err)
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m := readKustomizationFS(t, fSys)
|
m := readKustomizationFS(t, fSys)
|
||||||
for _, k := range []string{"patch1.yaml", "patch2.yaml"} {
|
for _, k := range []string{"patch1.yaml", "patch2.yaml"} {
|
||||||
if !patch.Exist(m.PatchesStrategicMerge, k) {
|
found := false
|
||||||
t.Errorf("%s must exist", k)
|
for _, p := range m.Patches {
|
||||||
|
if p.Path == k {
|
||||||
|
found = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
t.Fatalf("%s must exist", k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -108,30 +149,45 @@ func TestRemovePatchNotDefinedInKustomization(t *testing.T) {
|
|||||||
func TestRemovePatchNotExist(t *testing.T) {
|
func TestRemovePatchNotExist(t *testing.T) {
|
||||||
fSys := makeKustomizationPatchFS()
|
fSys := makeKustomizationPatchFS()
|
||||||
cmd := newCmdRemovePatch(fSys)
|
cmd := newCmdRemovePatch(fSys)
|
||||||
args := []string{"patch4.yaml"}
|
args := []string{
|
||||||
err := cmd.RunE(cmd, args)
|
"--path", "patch4.yaml",
|
||||||
|
"--kind", kind,
|
||||||
|
"--group", group,
|
||||||
|
"--version", version,
|
||||||
|
"--name", name,
|
||||||
|
"--namespace", namespace,
|
||||||
|
"--annotation-selector", annotationSelector,
|
||||||
|
"--label-selector", labelSelector,
|
||||||
|
}
|
||||||
|
cmd.SetArgs(args)
|
||||||
|
err := cmd.Execute()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("unexpected error %v", err)
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
m := readKustomizationFS(t, fSys)
|
m := readKustomizationFS(t, fSys)
|
||||||
for _, k := range []string{"patch1.yaml", "patch2.yaml"} {
|
for _, k := range []string{"patch1.yaml", "patch2.yaml"} {
|
||||||
if !patch.Exist(m.PatchesStrategicMerge, k) {
|
found := false
|
||||||
t.Errorf("%s must exist", k)
|
for _, p := range m.Patches {
|
||||||
|
if p.Path == k {
|
||||||
|
found = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
t.Fatalf("%s must exist", k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRemovePatchNoArgs(t *testing.T) {
|
func TestRemovePatchNoArgs(t *testing.T) {
|
||||||
|
// if no flags specified, we should do nothing
|
||||||
fSys := makeKustomizationPatchFS()
|
fSys := makeKustomizationPatchFS()
|
||||||
cmd := newCmdRemovePatch(fSys)
|
cmd := newCmdRemovePatch(fSys)
|
||||||
err := cmd.RunE(cmd, nil)
|
err := cmd.Execute()
|
||||||
|
|
||||||
if err == nil {
|
if err != nil {
|
||||||
t.Errorf("expected an error")
|
t.Fatalf("unexpected error %v", err)
|
||||||
}
|
|
||||||
if err.Error() != "must specify a patch file" {
|
|
||||||
t.Errorf("incorrect error: %v", err.Error())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user